try-with-resources and null resource
Joe Darcy
joe.darcy at oracle.com
Fri Jan 28 10:03:37 PST 2011
Gernot,
If you can share some statistics of the code base you looked at, lines
of code, # of finally block, # of finally blocks dealing with nullable,
resources, other informative things, I would find that to be an
interesting data point.
-Joe
Gernot Neppert wrote:
> Have you ever browsed a reasonably large codebase and counted the
> occurrences of 'finally' blocks that deal with nullable resources?
> Well, I have, and I found the infamous Class.getResourceAsStream to be
> the _only_ example of such a construct.
> And even that is easy to forego without adding any substantial
> clutter. Simply test the existence of the resource before opening the
> InputStream:
>
> URL resource = getClass().getResource("resourceThatMayOrMayNotExist");
> if(resource != null) {
> try(InputStream stream = resource.openStream())
> {
> ...
> }
> }
>
> _All other_ uses of try-finally blocks in my code deal with resources
> that are guaranteed to be non-null, because they
> are initialized by 'new' expressions or factory functions:
>
> java.sql.Statements, ResultSets, InputStreams, OutputStreams, Readers,
> Writers, Locks...
>
> In practice, having to deal with null-resources will probably be as
> rare as having to deal with null-iterators in a foreach loop.
>
>
> 2011/1/28 Reinier Zwitserloot <reinier at zwitserloot.com>:
>
>> I don't understand why one couldn't explain t-w-r by showing "if (resource
>> != null) resource.close();" in the finally block.
>>
>> It is then your opinion that this code is abusing the t-w-r construct,
>> because Class.getResourceAsStream can return null?
>>
>> try (InputStream introTextStream =
>> getClass().getResourceAsStream("intro.txt")) {
>> ...
>> }
>>
>> If so, we've clearly progressed into the "Everyone has their own 'feeling',
>> none of them agree with each other, and making language decisions based on
>> these feelings is a silly thing to do" phase. Let's check how often the
>> above occurs. I'm betting it'll be "loads of times". Also contrast what
>> you'd have to write if the above is not okay:
>>
>> InputStream introTextStream1 = getClass().getResourceAsStream();
>> if (introTextStream1 != null) {
>> try (InputStream introTextStream2 = introTextStream1) {
>> ...
>> }
>> }
>>
>> I assume I don't have to resort to 'feeling' to claim that the above is just
>> bad.
>>
>> --Reinier Zwitserloot
>>
>>
>>
>> On Fri, Jan 28, 2011 at 7:38 AM, David Holmes <David.Holmes at oracle.com>wrote:
>>
>>
>>> Tim Peierls said the following on 01/28/11 00:47:
>>>
>>>> I don't know about C#, but I like the formulation that can be sloppily
>>>> summarized as "t-w-r calls close on non-null resources".
>>>>
>>> So to summarize there are three potential candidates for:
>>>
>>> try (Resource r = getResource() {
>>> // stuff here
>>> }
>>>
>>> I'm ignoring the suppressed exception aspects for simplicity:
>>>
>>> a) Simple transformation into try-finally (current implemented semantics)
>>>
>>> Resource r = getResource();
>>> try {
>>> // stuff here
>>> }
>>> finally {
>>> r.close();
>>> }
>>>
>>> b) Transformation into try-finally with explicit null-check on entry
>>>
>>> Resource r = getResource();
>>> Objects.requireNonNull(r); // heh heh - couldn't resist!
>>> try {
>>> // stuff here
>>> }
>>> finally {
>>> r.close();
>>> }
>>>
>>> c) Simple transformation into try-finally but only close a non-null
>>> resource
>>>
>>> Resource r = getResource();
>>> try {
>>> // stuff here
>>> }
>>> finally {
>>> if (r != null)
>>> r.close();
>>> }
>>>
>>>
>>> Option (a) is, perhaps, the most consistent with preserving the
>>> behaviour of current code that would be modified to use the new
>>> construct. This construct will be taught by analogy as it is the only
>>> effective way to explain things "try-with-resources behaves as-if you
>>> had written ....". That said I think existing code would have an
>>> explicit null-check to avoid the try-finally in the null case. And I
>>> think it is potentially risky to execute the try block only to have NPE
>>> thrown later.
>>>
>>> Option (c) seems appealing in some sense because there's no NPE's at all
>>> if you don't use "r", but I think that appeal is superficial. To me the
>>> fundamental question to ask is: what is try-with-resources for? And I
>>> believe the answer to that is to initialize, use and then clean-up an
>>> AutoCloseable object. So to me using try-with-resource on a resource
>>> that might be null is simply a mis-use of try-with-resources: if it can
>>> be null you can't autoclose it, so trying to is a programming error and
>>> programming errors should be detected and reported as early as possible.
>>> So to me option (b) is the preferred semantics for this new construct.
>>>
>>> David Holmes
>>>
>>>
>>>
>>
>
>
More information about the coin-dev
mailing list