All Downloads are FREE. Search and download functionalities are using the official Maven repository.

javaslang.collection.Tree Maven / Gradle / Ivy

/*     / \____  _    _  ____   ______  / \ ____  __    _______
 *    /  /    \/ \  / \/    \ /  /\__\/  //    \/  \  //  /\__\   JΛVΛSLΛNG
 *  _/  /  /\  \  \/  /  /\  \\__\\  \  //  /\  \ /\\/ \ /__\ \   Copyright 2014-2016 Javaslang, http://javaslang.io
 * /___/\_/  \_/\____/\_/  \_/\__\/__/\__\_/  \_//  \__/\_____/   Licensed under the Apache License, Version 2.0
 */
package javaslang.collection;

import javaslang.Tuple;
import javaslang.Tuple2;
import javaslang.Tuple3;
import javaslang.collection.List.Nil;
import javaslang.collection.Tree.Empty;
import javaslang.collection.Tree.Node;
import javaslang.collection.TreeModule.*;
import javaslang.control.Option;

import java.io.*;
import java.util.*;
import java.util.function.*;
import java.util.stream.Collector;

import static javaslang.collection.Tree.Order.PRE_ORDER;

/**
 * A general Tree interface.
 *
 * @param  component type of this Tree
 * @author Daniel Dietrich, Grzegorz Piwowarek
 * @since 1.1.0
 */
public interface Tree extends Traversable {

    /**
     * Returns a {@link java.util.stream.Collector} which may be used in conjunction with
     * {@link java.util.stream.Stream#collect(java.util.stream.Collector)} to obtain a {@link javaslang.collection.Tree}.
     *
     * @param  Component type of the Tree.
     * @return A javaslang.collection.Tree Collector.
     */
    static  Collector, Tree> collector() {
        final Supplier> supplier = ArrayList::new;
        final BiConsumer, T> accumulator = ArrayList::add;
        final BinaryOperator> combiner = (left, right) -> {
            left.addAll(right);
            return left;
        };
        final Function, Tree> finisher = Tree::ofAll;
        return Collector.of(supplier, accumulator, combiner, finisher);
    }

    /**
     * Returns the singleton empty tree.
     *
     * @param  Type of tree values.
     * @return The empty tree.
     */
    static  Empty empty() {
        return Empty.instance();
    }

    /**
     * Narrows a widened {@code Tree} to {@code Tree}
     * by performing a type safe-cast. This is eligible because immutable/read-only
     * collections are covariant.
     *
     * @param tree An {@code Tree}.
     * @param   Component type of the {@code Tree}.
     * @return the given {@code tree} instance as narrowed type {@code Tree}.
     */
    @SuppressWarnings("unchecked")
    static  Tree narrow(Tree tree) {
        return (Tree) tree;
    }

    /**
     * Returns a new Node containing the given value and having no children.
     *
     * @param value A value
     * @param    Value type
     * @return A new Node instance.
     */
    static  Node of(T value) {
        return new Node<>(value, List.empty());
    }

    /**
     * Returns a new Node containing the given value and having the given children.
     *
     * @param value    A value
     * @param children The child nodes, possibly empty
     * @param       Value type
     * @return A new Node instance.
     */
    @SuppressWarnings("varargs")
    @SafeVarargs
    static  Node of(T value, Node... children) {
        Objects.requireNonNull(children, "children is null");
        return new Node<>(value, List.of(children));
    }

    /**
     * Returns a new Node containing the given value and having the given children.
     *
     * @param value    A value
     * @param children The child nodes, possibly empty
     * @param       Value type
     * @return A new Node instance.
     */
    static  Node of(T value, Iterable> children) {
        Objects.requireNonNull(children, "children is null");
        return new Node<>(value, List.ofAll(children));
    }

    /**
     * Creates a Tree of the given elements.
     *
     * @param     Component type of the List.
     * @param values Zero or more values.
     * @return A Tree containing the given values.
     * @throws NullPointerException if {@code values} is null
     */
    @SuppressWarnings("varargs")
    @SafeVarargs
    static  Tree of(T... values) {
        Objects.requireNonNull(values, "values is null");
        List list = List.of(values);
        return list.isEmpty() ? Empty.instance() : new Node<>(list.head(), list.tail().map(Tree::of));
    }

    /**
     * Creates a Tree of the given elements.
     * 

* If the given iterable is a tree, it is returned as result. * if the iteration order of the elements is stable. * * @param Component type of the List. * @param iterable An Iterable of elements. * @return A list containing the given elements in the same order. * @throws NullPointerException if {@code elements} is null */ @SuppressWarnings("unchecked") static Tree ofAll(Iterable iterable) { Objects.requireNonNull(iterable, "iterable is null"); if (iterable instanceof Tree) { return (Tree) iterable; } else { final List list = List.ofAll(iterable); return list.isEmpty() ? Empty.instance() : new Node<>(list.head(), list.tail().map(Tree::of)); } } /** * Returns a Tree containing {@code n} values of a given Function {@code f} * over a range of integer values from 0 to {@code n - 1}. * * @param Component type of the Tree * @param n The number of elements in the Tree * @param f The Function computing element values * @return A Tree consisting of elements {@code f(0),f(1), ..., f(n - 1)} * @throws NullPointerException if {@code f} is null */ static Tree tabulate(int n, Function f) { Objects.requireNonNull(f, "f is null"); return Collections.tabulate(n, f, Tree.empty(), Tree::of); } /** * Returns a Tree containing {@code n} values supplied by a given Supplier {@code s}. * * @param Component type of the Tree * @param n The number of elements in the Tree * @param s The Supplier computing element values * @return A Tree of size {@code n}, where each element contains the result supplied by {@code s}. * @throws NullPointerException if {@code s} is null */ static Tree fill(int n, Supplier s) { Objects.requireNonNull(s, "s is null"); return Collections.fill(n, s, Tree.empty(), Tree::of); } /** * Gets the value of this tree. * * @return The value of this tree. * @throws java.lang.UnsupportedOperationException if this tree is empty */ T getValue(); /** * Returns the children of this tree. * * @return the tree's children */ List> getChildren(); /** * Checks if this Tree is a leaf. A tree is a leaf if it is a Node with no children. * Because the empty tree is no Node, it is not a leaf by definition. * * @return true if this tree is a leaf, false otherwise. */ boolean isLeaf(); /** * Checks if this Tree is a branch. A Tree is a branch if it is a Node which has children. * Because the empty tree is not a Node, it is not a branch by definition. * * @return true if this tree is a branch, false otherwise. */ default boolean isBranch() { return !(isEmpty() || isLeaf()); } /** * Traverses this tree values in a specific {@link javaslang.collection.Tree.Order}. * * @param order A traversal order * @return A new Iterator */ default Iterator iterator(Order order) { return values(order).iterator(); } /** * Returns the number of nodes (including root and leafs). * * @return The size of the tree. */ int size(); /** * Transforms this {@code Tree}. * * @param f A transformation * @param Type of transformation result * @return An instance of type {@code U} * @throws NullPointerException if {@code f} is null */ default U transform(Function, ? extends U> f) { Objects.requireNonNull(f, "f is null"); return f.apply(this); } /** * Traverses this tree in {@link Order#PRE_ORDER}. * * @return A sequence of nodes. */ default Seq> traverse() { return traverse(PRE_ORDER); } /** * Traverses this tree in a specific order. * * @param order the tree traversal order * @return A sequence of nodes. * @throws java.lang.NullPointerException if order is null */ default Seq> traverse(Order order) { Objects.requireNonNull(order, "order is null"); if (isEmpty()) { return Stream.empty(); } else { final Node node = (Node) this; switch (order) { case PRE_ORDER: return Traversal.preOrder(node); case IN_ORDER: return Traversal.inOrder(node); case POST_ORDER: return Traversal.postOrder(node); case LEVEL_ORDER: return Traversal.levelOrder(node); default: throw new IllegalStateException("Unknown order: " + order.name()); } } } /** * Traverses this tree values in {@link Order#PRE_ORDER}. * Syntactic sugar for {@code traverse().map(Node::getValue)}. * * @return A sequence of the tree values. */ default Seq values() { return traverse(PRE_ORDER).map(Node::getValue); } /** * Traverses this tree values in a specific order. * Syntactic sugar for {@code traverse(order).map(Node::getValue)}. * * @param order the tree traversal order * @return A sequence of the tree values. * @throws java.lang.NullPointerException if order is null */ default Seq values(Order order) { return traverse(order).map(Node::getValue); } /** * Counts the number of branches of this tree. The empty tree and a leaf have no branches. * * @return The number of branches of this tree. */ default int branchCount() { if (isEmpty() || isLeaf()) { return 0; } else { return getChildren().foldLeft(1, (count, child) -> count + child.branchCount()); } } /** * Counts the number of leaves of this tree. The empty tree has no leaves. * * @return The number of leaves of this tree. */ default int leafCount() { if (isEmpty()) { return 0; } else if (isLeaf()) { return 1; } else { return getChildren().foldLeft(0, (count, child) -> count + child.leafCount()); } } /** * Counts the number of nodes (i.e. branches and leaves) of this tree. The empty tree has no nodes. * * @return The number of nodes of this tree. */ default int nodeCount() { if (isEmpty()) { return 0; } else { return 1 + getChildren().foldLeft(0, (count, child) -> count + child.nodeCount()); } } // -- Methods inherited from Traversable @Override default Seq distinct() { return values().distinct(); } @Override default Seq distinctBy(Comparator comparator) { Objects.requireNonNull(comparator, "comparator is null"); if (isEmpty()) { return Stream.empty(); } else { return values().distinctBy(comparator); } } @Override default Seq distinctBy(Function keyExtractor) { Objects.requireNonNull(keyExtractor, "keyExtractor is null"); if (isEmpty()) { return Stream.empty(); } else { return values().distinctBy(keyExtractor); } } @Override default Seq drop(long n) { if (n >= length()) { return Stream.empty(); } else { return values().drop(n); } } @Override default Seq dropRight(long n) { if (n >= length()) { return Stream.empty(); } else { return values().dropRight(n); } } @Override default Seq dropUntil(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); return dropWhile(predicate.negate()); } @Override default Seq dropWhile(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); if (isEmpty()) { return Stream.empty(); } else { return values().dropWhile(predicate); } } @Override default Seq filter(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); if (isEmpty()) { return Stream.empty(); } else { return values().filter(predicate); } } @Override default Tree flatMap(Function> mapper) { Objects.requireNonNull(mapper, "mapper is null"); return isEmpty() ? Empty.instance() : FlatMap.apply((Node) this, mapper); } @Override default U foldRight(U zero, BiFunction f) { Objects.requireNonNull(f, "f is null"); if (isEmpty()) { return zero; } else { return iterator().foldRight(zero, f); } } @SuppressWarnings("unchecked") @Override default Map> groupBy(Function classifier) { Objects.requireNonNull(classifier, "classifier is null"); if (isEmpty()) { return HashMap.empty(); } else { return (Map>) values().groupBy(classifier); } } @Override default Iterator> grouped(long size) { return sliding(size, size); } @Override default boolean hasDefiniteSize() { return true; } @Override default T head() { if (isEmpty()) { throw new NoSuchElementException("head of empty tree"); } else { return iterator().next(); } } @Override default Seq init() { if (isEmpty()) { throw new UnsupportedOperationException("init of empty tree"); } else { return values().init(); } } @Override default Option> initOption() { return isEmpty() ? Option.none() : Option.some(init()); } @Override default boolean isTraversableAgain() { return true; } @Override default Iterator iterator() { return values().iterator(); } @Override default int length() { return size(); } @Override default Tree map(Function mapper) { Objects.requireNonNull(mapper, "mapper is null"); return isEmpty() ? Empty.instance() : TreeModule.Map.apply((Node) this, mapper); } @SuppressWarnings("unchecked") @Override default Tuple2, Seq> partition(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); if (isEmpty()) { return Tuple.of(Stream.empty(), Stream.empty()); } else { return (Tuple2, Seq>) values().partition(predicate); } } @Override default Tree peek(Consumer action) { Objects.requireNonNull(action, "action is null"); if (!isEmpty()) { action.accept(head()); } return this; } @Override default Tree replace(T currentElement, T newElement) { if (isEmpty()) { return Empty.instance(); } else { return Replace.apply((Node) this, currentElement, newElement); } } @Override default Tree replaceAll(T currentElement, T newElement) { return map(t -> Objects.equals(t, currentElement) ? newElement : t); } @Override default Seq retainAll(Iterable elements) { Objects.requireNonNull(elements, "elements is null"); return values().retainAll(elements); } @Override default Seq scan(T zero, BiFunction operation) { return scanLeft(zero, operation); } @Override default Seq scanLeft(U zero, BiFunction operation) { Objects.requireNonNull(operation, "operation is null"); return Collections.scanLeft(this, zero, operation, List.empty(), List::prepend, List::reverse); } @Override default Seq scanRight(U zero, BiFunction operation) { Objects.requireNonNull(operation, "operation is null"); return Collections.scanRight(this, zero, operation, List.empty(), List::prepend, Function.identity()); } @Override default Iterator> sliding(long size) { return sliding(size, 1); } @Override default Iterator> sliding(long size, long step) { return iterator().sliding(size, step); } @SuppressWarnings("unchecked") @Override default Tuple2, Seq> span(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); if (isEmpty()) { return Tuple.of(Stream.empty(), Stream.empty()); } else { return (Tuple2, Seq>) values().span(predicate); } } @Override default Spliterator spliterator() { // the focus of the Stream API is on random-access collections of *known size* return Spliterators.spliterator(iterator(), length(), Spliterator.ORDERED | Spliterator.IMMUTABLE); } @Override default String stringPrefix() { return "Tree"; } @Override default Seq tail() { if (isEmpty()) { throw new UnsupportedOperationException("tail of empty tree"); } else { return values().tail(); } } @Override default Option> tailOption() { return isEmpty() ? Option.none() : Option.some(tail()); } @Override default Seq take(long n) { if (isEmpty()) { return Stream.empty(); } else { return values().take(n); } } @Override default Seq takeRight(long n) { if (isEmpty()) { return Stream.empty(); } else { return values().takeRight(n); } } @Override default Seq takeUntil(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); return values().takeUntil(predicate); } @Override default Seq takeWhile(Predicate predicate) { Objects.requireNonNull(predicate, "predicate is null"); return values().takeWhile(predicate); } @SuppressWarnings("unchecked") @Override default Tuple2, Tree> unzip( Function> unzipper) { Objects.requireNonNull(unzipper, "unzipper is null"); if (isEmpty()) { return Tuple.of(Empty.instance(), Empty.instance()); } else { return (Tuple2, Tree>) (Object) Unzip.apply((Node) this, unzipper); } } @SuppressWarnings("unchecked") @Override default Tuple3, Tree, Tree> unzip3( Function> unzipper) { Objects.requireNonNull(unzipper, "unzipper is null"); if (isEmpty()) { return Tuple.of(Empty.instance(), Empty.instance(), Empty.instance()); } else { return (Tuple3, Tree, Tree>) (Object) Unzip.apply3((Node) this, unzipper); } } @Override default Tree> zip(Iterable that) { Objects.requireNonNull(that, "that is null"); if (isEmpty()) { return Empty.instance(); } else { return Zip.apply((Node) this, that.iterator()); } } @Override default Tree> zipAll(Iterable that, T thisElem, U thatElem) { Objects.requireNonNull(that, "that is null"); if (isEmpty()) { return Iterator.ofAll(that).map(elem -> Tuple.of(thisElem, elem)).toTree(); } else { final java.util.Iterator thatIter = that.iterator(); final Tree> tree = ZipAll.apply((Node) this, thatIter, thatElem); if (thatIter.hasNext()) { final Iterable>> remainder = Iterator .ofAll(thatIter) .map(elem -> Tree.of(Tuple.of(thisElem, elem))); return new Node<>(tree.getValue(), tree.getChildren().appendAll(remainder)); } else { return tree; } } } @Override default Tree> zipWithIndex() { return zip(Iterator.from(0L)); } @Override boolean equals(Object o); @Override int hashCode(); @Override String toString(); /** * Creates a neat 2-dimensional drawing of a tree. Unicode characters are used to draw node junctions. * * @return A nice string representation of the tree. */ String draw(); /** * Represents a tree node. * * @param value type */ final class Node implements Tree, Serializable { private static final long serialVersionUID = 1L; private final T value; private final List> children; /** * Constructs a rose tree branch. * * @param value A value. * @param children A non-empty list of children. * @throws NullPointerException if children is null * @throws IllegalArgumentException if children is empty */ public Node(T value, List> children) { Objects.requireNonNull(children, "children is null"); this.value = value; this.children = children; } @Override public List> getChildren() { return children; } @Override public T getValue() { return value; } @Override public boolean isEmpty() { return false; } @Override public boolean isLeaf() { return children.isEmpty(); } @Override public int size() { return 1 + children.foldLeft(0, (acc, child) -> acc + child.length()); } @Override public boolean equals(Object o) { if (o == this) { return true; } else if (o instanceof Node) { final Node that = (Node) o; return Objects.equals(this.getValue(), that.getValue()) && Objects.equals(this.getChildren(), that.getChildren()); } else { return false; } } @Override public int hashCode() { return Objects.hash(value, children); } @Override public String toString() { return stringPrefix() + (isLeaf() ? "(" + value + ")" : toLispString(this)); } @Override public String draw() { StringBuilder builder = new StringBuilder(); drawAux("", builder); return builder.toString(); } private void drawAux(String indent, StringBuilder builder) { builder.append(value); for (List> it = children; !it.isEmpty(); it = it.tail()) { final boolean isLast = it.tail().isEmpty(); builder.append('\n') .append(indent) .append(isLast ? "└──" : "├──"); it.head().drawAux(indent + (isLast ? " " : "│ "), builder); } } private static String toLispString(Tree tree) { final String value = String.valueOf(tree.getValue()); if (tree.isLeaf()) { return value; } else { return String.format("(%s %s)", value, tree.getChildren().map(Node::toLispString).mkString(" ")); } } // -- Serializable implementation /** * {@code writeReplace} method for the serialization proxy pattern. *

* The presence of this method causes the serialization system to emit a SerializationProxy instance instead of * an instance of the enclosing class. * * @return A SerialiationProxy for this enclosing class. */ private Object writeReplace() { return new SerializationProxy<>(this); } /** * {@code readObject} method for the serialization proxy pattern. *

* Guarantees that the serialization system will never generate a serialized instance of the enclosing class. * * @param stream An object serialization stream. * @throws java.io.InvalidObjectException This method will throw with the message "Proxy required". */ private void readObject(ObjectInputStream stream) throws InvalidObjectException { throw new InvalidObjectException("Proxy required"); } /** * A serialization proxy which, in this context, is used to deserialize immutable nodes with final * instance fields. * * @param The component type of the underlying tree. */ // DEV NOTE: The serialization proxy pattern is not compatible with non-final, i.e. extendable, // classes. Also, it may not be compatible with circular object graphs. private static final class SerializationProxy implements Serializable { private static final long serialVersionUID = 1L; // the instance to be serialized/deserialized private transient Node node; /** * Constructor for the case of serialization, called by {@link Node#writeReplace()}. *

* The constructor of a SerializationProxy takes an argument that concisely represents the logical state of * an instance of the enclosing class. * * @param node a Branch */ SerializationProxy(Node node) { this.node = node; } /** * Write an object to a serialization stream. * * @param s An object serialization stream. * @throws java.io.IOException If an error occurs writing to the stream. */ private void writeObject(ObjectOutputStream s) throws IOException { s.defaultWriteObject(); s.writeObject(node.value); s.writeObject(node.children); } /** * Read an object from a deserialization stream. * * @param s An object deserialization stream. * @throws ClassNotFoundException If the object's class read from the stream cannot be found. * @throws IOException If an error occurs reading from the stream. */ @SuppressWarnings("unchecked") private void readObject(ObjectInputStream s) throws ClassNotFoundException, IOException { s.defaultReadObject(); final T value = (T) s.readObject(); final List> children = (List>) s.readObject(); node = new Node<>(value, children); } /** * {@code readResolve} method for the serialization proxy pattern. *

* Returns a logically equivalent instance of the enclosing class. The presence of this method causes the * serialization system to translate the serialization proxy back into an instance of the enclosing class * upon deserialization. * * @return A deserialized instance of the enclosing class. */ private Object readResolve() { return node; } } } /** * The empty tree. Use Tree.empty() to create an instance. * * @param type of the tree's values */ final class Empty implements Tree, Serializable { private static final long serialVersionUID = 1L; private static final Empty INSTANCE = new Empty<>(); // hidden private Empty() { } @SuppressWarnings("unchecked") public static Empty instance() { return (Empty) INSTANCE; } @Override public List> getChildren() { return Nil.instance(); } @Override public T getValue() { throw new UnsupportedOperationException("getValue of empty Tree"); } @Override public boolean isEmpty() { return true; } @Override public boolean isLeaf() { return false; } @Override public int size() { return 0; } @Override public boolean equals(Object o) { return o == this; } @Override public int hashCode() { return 1; } @Override public String toString() { return stringPrefix() + "()"; } @Override public String draw() { return "▣"; } // -- Serializable implementation /** * Instance control for object serialization. * * @return The singleton instance of Nil. * @see java.io.Serializable */ private Object readResolve() { return INSTANCE; } } /** * Tree traversal order. *

* Example tree: *

     * 
     *         1
     *        / \
     *       /   \
     *      /     \
     *     2       3
     *    / \     /
     *   4   5   6
     *  /       / \
     * 7       8   9
     * 
     * 
* * See also * */ // see http://programmers.stackexchange.com/questions/138766/in-order-traversal-of-m-way-trees enum Order { /** * 1 2 4 7 5 3 6 8 9 (= depth-first) */ PRE_ORDER, /** * 7 4 2 5 1 8 6 9 3 */ IN_ORDER, /** * 7 4 5 2 8 9 6 3 1 */ POST_ORDER, /** * 1 2 3 4 5 6 7 8 9 (= breadth-first) */ LEVEL_ORDER } } /** * Because the empty tree {@code Empty} cannot be a child of an existing tree, method implementations distinguish between the * empty and non-empty case. Because the structure of trees is recursive, often we have commands in the form of module * classes with one static method. */ interface TreeModule { final class FlatMap { @SuppressWarnings("unchecked") static Tree apply(Node node, Function> mapper) { final Tree mapped = Tree.ofAll(mapper.apply(node.getValue())); if (mapped.isEmpty()) { return Tree.empty(); } else { final List> children = (List>) (Object) node .getChildren() .map(child -> FlatMap.apply(child, mapper)) .filter(Tree::nonEmpty); return Tree.of(mapped.getValue(), children.prependAll(mapped.getChildren())); } } } final class Map { static Node apply(Node node, Function mapper) { final U value = mapper.apply(node.getValue()); final List> children = node.getChildren().map(child -> Map.apply(child, mapper)); return new Node<>(value, children); } } final class Replace { // Idea: // Traverse (depth-first) until a match is found, then stop and rebuild relevant parts of the tree. // If not found, return the same tree instance. static Node apply(Node node, T currentElement, T newElement) { if (Objects.equals(node.getValue(), currentElement)) { return new Node<>(newElement, node.getChildren()); } else { for (Node child : node.getChildren()) { final Node newChild = Replace.apply(child, currentElement, newElement); final boolean found = newChild != child; if (found) { final List> newChildren = node.getChildren().replace(child, newChild); return new Node<>(node.getValue(), newChildren); } } return node; } } } final class Traversal { static Stream> preOrder(Node node) { return node.getChildren().foldLeft(Stream.of(node), (acc, child) -> acc.appendAll(preOrder(child))); } static Stream> inOrder(Node node) { if (node.isLeaf()) { return Stream.of(node); } else { final List> children = node.getChildren(); return children .tail() .foldLeft(Stream.> empty(), (acc, child) -> acc.appendAll(inOrder(child))) .prepend(node) .prependAll(inOrder(children.head())); } } static Stream> postOrder(Node node) { return node .getChildren() .foldLeft(Stream.> empty(), (acc, child) -> acc.appendAll(postOrder(child))) .append(node); } static Stream> levelOrder(Node node) { Stream> result = Stream.empty(); final java.util.Queue> queue = new java.util.LinkedList<>(); queue.add(node); while (!queue.isEmpty()) { final Node next = queue.remove(); result = result.prepend(next); queue.addAll(next.getChildren().toJavaList()); } return result.reverse(); } } final class Unzip { static Tuple2, Node> apply(Node node, Function> unzipper) { final Tuple2 value = unzipper.apply(node.getValue()); final List, Node>> children = node .getChildren() .map(child -> Unzip.apply(child, unzipper)); final Node node1 = new Node<>(value._1, children.map(t -> t._1)); final Node node2 = new Node<>(value._2, children.map(t -> t._2)); return Tuple.of(node1, node2); } static Tuple3, Node, Node> apply3(Node node, Function> unzipper) { final Tuple3 value = unzipper.apply(node.getValue()); final List, Node, Node>> children = node.getChildren() .map(child -> Unzip.apply3(child, unzipper)); final Node node1 = new Node<>(value._1, children.map(t -> t._1)); final Node node2 = new Node<>(value._2, children.map(t -> t._2)); final Node node3 = new Node<>(value._3, children.map(t -> t._3)); return Tuple.of(node1, node2, node3); } } final class Zip { @SuppressWarnings("unchecked") static Tree> apply(Node node, java.util.Iterator that) { if (!that.hasNext()) { return Empty.instance(); } else { final Tuple2 value = Tuple.of(node.getValue(), that.next()); final List>> children = (List>>) (Object) node .getChildren() .map(child -> Zip.apply(child, that)) .filter(Tree::nonEmpty); return new Node<>(value, children); } } } final class ZipAll { @SuppressWarnings("unchecked") static Tree> apply(Node node, java.util.Iterator that, U thatElem) { if (!that.hasNext()) { return node.map(value -> Tuple.of(value, thatElem)); } else { final Tuple2 value = Tuple.of(node.getValue(), that.next()); final List>> children = (List>>) (Object) node .getChildren() .map(child -> ZipAll.apply(child, that, thatElem)) .filter(Tree::nonEmpty); return new Node<>(value, children); } } } }