Revisiting MayHoldCloseableResource

Joshua Bloch josh at bloch.us
Tue Aug 13 01:03:12 PDT 2013


Having BaseStream implement AutoCloseable because some streams need to be
closed seems like a bad idea.

     Josh


On Mon, Aug 12, 2013 at 10:43 AM, Brian Goetz <brian.goetz at oracle.com>wrote:

> After discussion both in the EG and within Oracle, here's where I think
> this should land:
>
>  - Eliminate MayHoldCloseableResource, which is fundamentally flawed
> unless the spec for AutoCloseable is fixed, in which case it becomes
> unnecessary.
>  - Fix the spec for AutoCloseable.
>  - Eliminate @HoldsResource, not because it is intrinsically bad, but
> because it is a half solution and we should work towards a complete one for
> 9, likely using something like JSR-308 checkers.  The Eclipse inspector can
> be adjusted to work with the new APIs.
>  - Make BaseStream implement AutoCloseable, with additional verbiage
> regarding closing.
>  - Add additional spec to Files.{walk,lines,etc}, and to combinators like
> concat, regarding their close behavior.
>
> The thing that pushed me over the edge from the various "work around the
> problem" solutions is the existence of flatMap:
>
>   Files.walk(dir)
>        .flatMap(Files::lines)
>        ...
>
> where each entry resulting from walk() will turn into a stream creation
> and traversal within the flatMap implementation, and flatMap itself needs a
> way to ensure it is deterministically closed.  This pushes strongly for
> having a close() method on Stream, which, if you're going to have that, you
> might as well be AutoCloseable to make life easier for users.
>
> The sole challenge here is to keep people from thinking that they *need*
> to close all streams, and therefore mangle their code in bad ways to
> satisfy this perceived need.
>
>
>
> On 8/2/2013 7:39 PM, Brian Goetz wrote:
>
>> There seems to be consensus that the current strawman for
>> MayHoldCloseableResoruce and @HoldsResource is too flawed to proceed.
>>
>> Flaw #1 relates to @HoldsResource.  While there seems to be consensus
>> that annotations are reasonable for imparting information like this which
>> is suited to static analysis tools, the current proposal (a) confusingly
>> leaks through into subclass Javadoc and (b) seems too weak.  Conclusion:
>> just dropping @HR from the current scheme (leaving only MHCR <: AC) is an
>> improvement; we can revisit what the correct annotations for hinting at
>> resourceful-ness later.
>>
>> Flaw #2 relates to MHCR.  The real problem is that AutoCloseable is
>> overspecified; it says "a resource that must be closed."  This is an
>> overreach; what j.l.AC should say is "has a close method which is called
>> automatically on exit from a TWR block."  Because of this overspecify,
>> subclassing doesn't help, because if an AC must be closed, so must a MHCR.
>>  Conclusion: dropping MHCR and just having BaseStream <: AC is an
>> improvement.
>>
>> Flaw #3 relates to the spec of AC.  The conclusion of the core library
>> maintainers is that it is acceptable to correct the spec, and replace the
>> "must" with "may".  This leaves us with a fixed AC, and BaseStream <: AC.
>>  This seems strictly better than where we are now.  Let's call this "Option
>> 1."
>>
>> But, it is possible we might still choose to punt further for now.
>>
>> Observation: the need for Stream to have a close() method comes about
>> because of the desire to have static factory methods that will create
>> streams that encapsulate a resource that requires closing, when there is no
>> way to get access to that underlying resource.  Such as Files.walk(dir) or
>> Files.lines(path).
>>
>> For other cases, like BufferedReader.lines(), the BR itself is AC, so
>> users can handle resource management like:
>>
>>    try (BufferedReader b = ...) {
>>       b.lines()...
>>    }
>>
>> and there is no need to close the *stream* at all, since the user has
>> access to the actual closeable resource.  It is only the static methods
>> such as Files.list(dir), which don't provide the user access in this way.
>>  The methods we provide are:
>>
>> Files.list(Path)
>> Files.walk(Path, maxDepth, options)
>> Files.walk(Path, options)
>> Files.find(Path, maxDepth, BiPredicate<Path, BasicFileAttributes>,
>> options)
>> Files.lines(Path)
>>
>> Colin @ Google pointed out that if the spliterators for these streams
>> were lazy (didn't open the resource until the terminal began), and all
>> terminal ops called the close hook in a finally block after processing,
>> then we wouldn't need to be AC, since there would be no way to leak a
>> closeable resource.  (This would move resource-not-found exceptions from
>> the factory method to the terminal op.)
>>
>> Still, having a close() method and *not* being AC may look weird to the
>> users.
>>
>> Let's call this "Option 2": have a close() method, don't extend AC, make
>> sure that all terminals call close, make sure that all static
>> resource-holding stream factories produce late-binding spliterators.
>>
>> Option 3 would be to simply get rid of these methods (note that
>> Files.list and Files.lines are pretty simple wrappers around existing AC
>> abstractions, DirectoryStream and BufferedReader.)
>>
>> Orthogonal to Options 2 and 3 would be fixing the spec of AC.
>>
>>
>>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: http://mail.openjdk.java.net/pipermail/lambda-libs-spec-experts/attachments/20130813/ad385aaa/attachment.html 


More information about the lambda-libs-spec-experts mailing list