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());
}
由于 Stream
是 BaseStream
的子类,并且 BaseStream
扩展了 AutoClosable
,您应该可以使用 try-with-resource
模式。
如果在调用构建的流链的任何元素期间发生异常,是否有关闭 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());
}
由于 Stream
是 BaseStream
的子类,并且 BaseStream
扩展了 AutoClosable
,您应该可以使用 try-with-resource
模式。