我被要求检索作为树节点的 descandant 的每个叶节点。我很快就想到我可以在一条线上完成这项工作!
public Set<TreeNode<E>> getLeaves() {
return getChildrenStream().flatMap(n -> n.getChildrenStream()).collect(toSet());
}
乍一看很好,但StackOverflowExcepetion
如果树深度达到〜10,它很快就遇到了,这是我无法接受的。后来我开发了一个没有递归和流的实现(但是我的大脑被烤了),但我仍然想知道是否有办法flatMap
用流做递归,因为我发现如果不接触流内部就不可能这样做。它需要一个新的操作,就像RecursiveOps
这样做,否则我必须将所有结果收集到Set
每一步中,然后再对其进行操作Set
:
Set<TreeNode<E>> prev = new HashSet<>();
prev.add(this);
while (!prev.isEmpty()) {
prev = prev.stream().flatMap(n -> n.getChildrenStream()).collect(toSet());
}
return prev;
不像看起来那么好。流意味着是一个管道。在添加终端操作之前,不会计算其结果和中间结果。上述做法显然违反了这一原则。并行化也不像流那么容易。我可以在不手动计算所有中间结果的情况下递归 flatMap 吗?
PS1:TreeNode 声明:
public class TreeNode<E> {
// ...
/**
* Get a stream of children of the current node.
*
*/
public Stream<TreeNode<E>> getChildrenStream(){
// ...
}
public Set<TreeNode<E>> getLeaves() {
// main concern
}
}f