Java 8 - 出现异常时关闭流?

Java 8 - closing stream on Exception?

如果在调用构建的流链的任何元素期间发生异常,是否有关闭 Stream 的方法?

例如代码:

    Stream.of(new Object())
            .filter(e -> {
                if (true) throw new IllegalStateException();
                return true;
            })
            .onClose(() -> System.out.println("OnClose"))
            .collect(Collectors.toList());

将产生以下输出:

Exception in thread "main" java.lang.IllegalStateException
at my.className.lambda$main(LdapRealm.java:114)
at my.className$$Lambda/1607521710.test(Unknown Source)
at java.util.stream.ReferencePipeline.accept(ReferencePipeline.java:174)
at java.util.stream.Streams$StreamBuilderImpl.forEachRemaining(Streams.java:419)
at java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:512)
at java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:502)
at java.util.stream.ReduceOps$ReduceOp.evaluateSequential(ReduceOps.java:708)
at java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:234)
at java.util.stream.ReferencePipeline.collect(ReferencePipeline.java:499)
at my.className.main(LdapRealm.java:118)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

请注意,"OnClose" 消息未打印,因为尚未调用 onClose()。如果我想让这个流被第三方代码使用,我不确定这个代码是否会促进 try/catch+Autocloseable 功能,有什么好的解决方案吗?

如果您的 class 的用户不使用 try/finally 或 try-with-resources,您绝对不能做任何清理工作。 (终结器除外,这不是一个好的解决方案。)

如果你有一个函数 returns 流

Stream<Object> getStream() {
    return Stream.of(new Object())
        .filter(e -> {
            if (true) throw new IllegalStateException();
            return true;
        })
        .onClose(() -> System.out.println("OnClose"));
}

那么这个函数的正确用法是

List<Object> result;
try (Stream<Object> s = getStream()) {
    result = s.collect(Collectors.toList());
}

由于 StreamBaseStream 的子类,并且 BaseStream 扩展了 AutoClosable,您应该可以使用 try-with-resource 模式。