<div dir="ltr">I like the benefits, though I think the onClose name is confusing.<div><br></div><div style>In many frameworks, the onError/onExit/onEvent naming style is reserved for method names in callback interfaces.  It is the prevalent convention currently.  So I&#39;d expect onClose() to be the method that is called-back upon closing.</div>
<div style><br></div><div style>The method name that is needed for this wrapping purpose should be something more imperative, possibly mentioning its function as a &quot;close&quot; guard.</div><div style><br></div><div style>
Some ideas below:</div><div style><br></div><div style>.autoclose(ac)</div><div style>.autoclosing(ac)</div><div style><br></div><div style>.closeableStream(ac)</div><div style><br></div><div style>--Joe</div></div><div class="gmail_extra">
<br><br><div class="gmail_quote">On Thu, Jun 20, 2013 at 8:22 AM, Brian Goetz <span dir="ltr">&lt;<a href="mailto:brian.goetz@oracle.com" target="_blank">brian.goetz@oracle.com</a>&gt;</span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Paul, this looks great.  I love that we can get rid of CloseableStream and DelegatingStream (two of the last remaining top-level API warts), and efficiently incorporate resource release in a predictable manner.<div class="HOEnZb">
<div class="h5"><br>
<br>
<br>
<br>
<br>
On 6/20/2013 11:17 AM, Paul Sandoz wrote:<br>
<blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Hi,<br>
<br>
Having another go at this...<br>
<br>
The current solution of resource management for Stream is poor (as previously mentioned):<br>
<br>
1) CloseableStream/<u></u>DelegatingStream add a lot of API surface area.<br>
<br>
2) Operations on CloseableStream do not return a CloseableStream.<br>
<br>
3) try-with-resources always needs to be used with the CloseableStream since the Stream itself has no closeable semantics.<br>
<br>
4) Stream.concat fails to propagate the closing<br>
<br>
<br>
If Stream extends from AutoCloseable we can address the first 3 issues. In fact it can be addressed with just a close method but it is awkward to transform that into an AutoCloseable for use with try-with-resources: try (AutoCloseable ac = () -&gt; s.close()) { … }<br>

<br>
A negative point is it is no longer clear whether a stream should be closed or not, which is anyway the case for issues 2/3/4. However, i don&#39;t think that should stop us trying to improve the general situation, it&#39;s not gonna be perfect but i think we can do better than what we currently have [*].<br>

<br>
<br>
Issue 4 can be addressed by adding a Stream.onClose(AutoCloseable ac) method.<br>
<br>
Stream s = ...<br>
s = s.onClose(a).filter(...).<u></u>onClose(b).<br>
s.close(); // b is called, then a is called<br>
s.toArray(); // throws ISE<br>
<br>
Stream s = ...<br>
s = s.onClose(a).onClose(b).<br>
s.close(); // b is called, then a is called<br>
s.toArray(); // throws ISE<br>
<br>
The Stream.concat implementation becomes:<br>
<br>
         Stream&lt;T&gt; cs = (a.isParallel() || b.isParallel())<br>
                ? StreamSupport.parallelStream(<u></u>split)<br>
                : StreamSupport.stream(split);<br>
<br>
         return cs.onClose(() -&gt; { a.close(); b.close(); } ) // ignoring exception handling to be brief<br>
<br>
<br>
The Stream.close/onClose methods enable us to specify more precisely the behaviour of Stream.close, the order in which calls to close on AutoClosable instances passed to onClose occur, and what happens if AutoCloseable.close throws an exception.<br>

<br>
<br>
Of course it is possible to do silly things like this:<br>
<br>
   s = s.conClose(a).filter(...).<u></u>onClose(s).<br>
<br>
but we could detect if s is stage in the pipeline and throw an IAE.<br>
<br>
<br>
FWIW Stream.close/onClose is easy to implement efficiently.<br>
<br>
Paul.<br>
<br>
[*] The JDK world is murky, see ByteArrayInputStream.close:<br>
<br>
     /**<br>
      * Closing a &lt;tt&gt;ByteArrayInputStream&lt;/tt&gt; has no effect. The methods in<br>
      * this class can be called after the stream has been closed without<br>
      * generating an &lt;tt&gt;IOException&lt;/tt&gt;.<br>
      * &lt;p&gt;<br>
      */<br>
     public void close() throws IOException {<br>
     }<br>
<br>
</blockquote>
</div></div></blockquote></div><br></div>