Java标记的联合/和类型
有什么方法可以在Java中定义求和类型?Java似乎自然地直接支持产品类型,我认为枚举可能允许它支持求和类型,而继承看起来也许可以做到,但是至少有一种情况我无法解决。详细地说,求和类型是可以恰好具有一组不同类型之一的类型,例如C中的标记联合。就我而言,我正在尝试在Java中实现haskell的Either类型:
data Either a b = Left a | Right b
但在基本级别上,我必须将其实现为产品类型,而忽略其字段之一:
public class Either<L,R>{
private L left = null;
private R right = null;
public static <L,R> Either<L,R> right(R right)
{
return new Either<>(null, right);
}
public static <L,R> Either<L,R> left(L left)
{
return new Either<>(left, null);
}
private Either(L left, R right) throws IllegalArgumentException
{
this.left = left;
this.right = right;
if (left != null && right != null)
{
throw new IllegalArgumentException("An Either cannot be created with two values");
}
if (left == right)
{
throw new IllegalArgumentException("An Either cannot be created without a value");
}
}
.
.
.
}
我尝试通过继承来实现,但是我必须使用通配符类型参数或等效参数,Java泛型不允许使用:
public class Left<L> extends Either<L,?>
我没有太多使用Java的Enums,但是尽管它们似乎是次佳的候选者,但我并不抱有希望。
在这一点上,我认为只有通过类型转换Object
值才有可能做到这一点,我希望完全避免这种情况,除非有一种方法可以安全,一次性地对所有求和类型使用它。
回答:
制作Either
一个没有字段且只有一个构造函数(私有,无参数,空)的抽象类,并将您的“数据构造函数”(left
和right
静态工厂方法)嵌套在该类中,以便他们可以看到私有构造函数,而其他则不能有效密封型。
使用抽象方法either
来模拟详尽的模式匹配,并适当地覆盖静态工厂方法返回的具体类型。实现方便的方法(如fromLeft
,fromRight
,bimap
,first
,second
而言)either
。
import java.util.Optional;import java.util.function.Function;
public abstract class Either<A, B> {
private Either() {}
public abstract <C> C either(Function<? super A, ? extends C> left,
Function<? super B, ? extends C> right);
public static <A, B> Either<A, B> left(A value) {
return new Either<A, B>() {
@Override
public <C> C either(Function<? super A, ? extends C> left,
Function<? super B, ? extends C> right) {
return left.apply(value);
}
};
}
public static <A, B> Either<A, B> right(B value) {
return new Either<A, B>() {
@Override
public <C> C either(Function<? super A, ? extends C> left,
Function<? super B, ? extends C> right) {
return right.apply(value);
}
};
}
public Optional<A> fromLeft() {
return this.either(Optional::of, value -> Optional.empty());
}
}
愉快而安全!没有办法把它搞砸。因为类型是有效密封的,所以您可以放心,只有两种情况,并且最终每个操作都必须根据either
方法进行定义,这将迫使调用者处理这两种情况。
关于您试图解决的问题class Left<L> extends Either<L,?>
,请考虑签名<A, B> Either<A, B> left(A
value)。类型参数B
未出现在参数列表中。因此,考虑到某种类型的值A
,你可以得到一个Either<A, B>
对 任何 类型B
。
以上是 Java标记的联合/和类型 的全部内容, 来源链接: utcz.com/qa/397994.html