Реализовать Декартовое произведение коллекций Java 8
прямо сейчас я могу реализовать только декартово произведение двух коллекций, вот код:
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, но не затмение (оба с уровнем соответствия компилятора 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>)
вот пара.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
)
2 ответов
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
вот решение, которое обобщает случай, когда количество требуемых flatMap-приложений (i. e. порядок продукта) не известен во время компиляции.
BinaryOperator<Function<String,Stream<String>>> kleisli = (f,g) -> s -> f.apply(s).flatMap(g);
List<String> cartesian(int n, Collection<String> coll) {
return coll.stream()
.flatMap( IntStream.range(1, n).boxed()
.map(_any -> crossWith(coll::stream, String::concat)) // create (n-1) appropriate crossWith instances
.reduce(s -> Stream.of(s), kleisli) // compose them sequentially
) // flatMap the stream with the entire function chain
.collect(toList());
}
вы найдете подробную информацию о том, как это работает в мой блог.