带有instanceof和类cast的方法引用的Java流

时间:2017-05-24 12:34:00

标签: java java-8 java-stream method-reference

我可以使用方法参考转换以下代码吗?

List<Text> childrenToRemove = new ArrayList<>();

group.getChildren().stream()
    .filter(c -> c instanceof Text)
    .forEach(c -> childrenToRemove.add((Text)c));

让我举一个例子来说明我的意思,假设我们有

myList
    .stream()
    .filter(s -> s.startsWith("c"))
    .map(String::toUpperCase)
    .sorted()
    .forEach(elem -> System.out.println(elem));

使用方法参考它可以写成(最后一行)

myList
    .stream()
    .filter(s -> s.startsWith("c"))
    .map(String::toUpperCase)
    .sorted()
    .forEach(System.out::println);

将表达式转换为方法引用有哪些规则?

3 个答案:

答案 0 :(得分:17)

是的,您可以使用以下方法参考:

.filter(Text.class::isInstance)
.map(Text.class::cast)
.forEach(childrenToRemove::add);

您可以使用Collectors.toSet()收集流项:

,而不是for-each-add
Set<Text> childrenToRemove = group.getChildren()
    // ...
    .collect(Collectors.toSet());

如果您需要维护孩子的顺序,请使用toList()

如果签名匹配,则可以使用方法引用替换lambda表达式:

ContainingClass::staticMethodName // method reference to a static method
containingObject::instanceMethodName // method reference to an instance method
ContainingType::methodName // method reference to an instance method
ClassName::new // method reference to a constructor

答案 1 :(得分:0)

我认为是可能的,就像这样

group.getChildren()
    .filter(Text.class::isInstance)
    .map(Text.class::cast)
    .collect(Collectors.toCollection(() -> childrenToRemove));

答案 2 :(得分:0)

很抱歉,该线程已恢复,但最后一个答案不符合预期的行为。通过阅读Java文档,供应商将返回一个空集合。这很重要,因为在收集过程中可能会多次致电供应商。 例如:

...collect(Collectors.toCollection(ArrayList::new));

对Java文档的引用:JavaDoc