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一个没有字段且只有一个构造函数(私有,无参数,空)的抽象类,并将您的“数据构造函数”(leftright静态工厂方法)嵌套在该类中,以便他们可以看到私有构造函数,而其他则不能有效密封型。

使用抽象方法either来模拟详尽的模式匹配,并适当地覆盖静态工厂方法返回的具体类型。实现方便的方法(如fromLeftfromRightbimapfirstsecond而言)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

回到顶部