gpt4 book ai didi

java - 通过Java Stream实现多个集合的笛卡尔积

转载 作者:行者123 更新时间:2023-12-02 07:54:17 33 4
gpt4 key购买 nike

现在我只能实现两个集合的笛卡尔积,代码如下:

public static <T1, T2, R extends Collection<Pair<T1, T2>>>
R getCartesianProduct(
Collection<T1> c1, Collection<T2> c2,
Collector<Pair<T1, T2>, ?, R> collector) {
return c1.stream()
.flatMap(e1 -> c2.stream().map(e2 -> new Pair<>(e1, e2)))
.collect(collector);
}

此代码在 IntelliJ 中运行良好,但在 Eclipse 中运行不佳。两者的编译器合规级别均为 1.8:

The method collect(Collector<? super Object,A,R>)
in the type Stream<Object> is not applicable for
the arguments (Collector<Pair<T1,T2>,capture#5-of ?,R>)

这是 Pair.java:

public class Pair<T1, T2> implements Serializable {
protected T1 first;
protected T2 second;
private static final long serialVersionUID = 1360822168806852921L;

public Pair(T1 first, T2 second) {
this.first = first;
this.second = second;
}

public Pair(Pair<T1, T2> pair) {
this(pair.getFirst(), pair.getSecond());
}

public T1 getFirst() {
return this.first;
}

public T2 getSecond() {
return this.second;
}

public void setFirst(T1 o) {
this.first = o;
}

public void setSecond(T2 o) {
this.second = o;
}

public String toString() {
return "(" + this.first + ", " + this.second + ")";
}

@Override
public boolean equals(Object o) {
if(!(o instanceof Pair))
return false;
Pair p = (Pair) o;
if(!this.first.equals(p.first))
return false;
if(!this.second.equals(p.second))
return false;
return true;
}

@Override
public int hashCode() {
int hash = 1;
hash = hash * 31 + this.first.hashCode();
hash = hash * 31 + this.second.hashCode();
return hash;
}
}

如何修复这个错误?

是否有一种优雅的方式来实现多个集合的笛卡尔积?假设我们有类tuple

最佳答案

Eclipse 存在类型推断问题。如果添加类型提示.<Pair<T1,T2>>flatMap ,它编译得很好。

如果我建议采用不同的方法,请考虑让您的 cartesianProduct 不执行整个流和集合,而只是 flatMap 的帮助者。 :

static <T1, T2, R> Function<T1, Stream<R>> crossWith(
Supplier<? extends Stream<T2>> otherSup,
BiFunction<? super T1, ? super T2, ? extends R> combiner
) {
return t1 -> otherSup.get().map(t2 -> combiner.apply(t1, t2));
}

现在你只需要创建一个 Pair如果您希望结果包含 Pair s,您可以通过应用 flatMap 进行高阶笛卡尔积几次:

List<String> letters = Arrays.asList("A", "B", "C");
List<Integer> numbers = Arrays.asList(1, 2, 3);

List<Pair<String, Integer>> board = letters.stream()
.flatMap(crossWith(numbers::stream, Pair::new))
.collect(toList());


List<String> ops = Arrays.asList("+", "-", "*", "/");

List<String> combinations = letters.stream()
.flatMap(crossWith(ops::stream, String::concat))
.flatMap(crossWith(letters::stream, String::concat))
.collect(toList()); // triple cartesian product

关于java - 通过Java Stream实现多个集合的笛卡尔积,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31374411/

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com