Common try-with-resources Idiom Won't Clean Up Properly
Join the DZone community and get the full member experience.
Join For FreeSo I was re-reading some blogs on the JDK 7 language changes and I noticed the following idiom seems popular:
// try (BufferedReader br = new BufferedReader( new FileReader("/tmp/click.xml"))) { System.out.println(br.readLine()); }
That seems all well and good; but consider what happens should something
goes wrong constructing the BufferedReader. The contract for
try-with-resources will only try to close and tidy up the declared
field, if something goes wrong then FileReader is never closed.
So consider this example where the constructor of BufferedReader throws
an Error, it could well be an OutOfMemoryError or any number of other
failure modes.
// package client; import java.io.*; public class BufferedFileExample { public static void main(String[] args) throws FileNotFoundException, IOException { try (BufferedReader br = new MyBufferedReader( new MyFileReader("/tmp/click.xml"))) { System.out.println(br.readLine()); } } public static class MyFileReader extends FileReader { public MyFileReader(String in) throws FileNotFoundException { super(in); } public void close() throws IOException { super.close(); System.out.println("Close called"); } } public static class MyBufferedReader extends BufferedReader { public MyBufferedReader(Reader in) { super(in); throw new Error(); } } }
This gets us the following output and as such the close is not performed
on the FileReader which until the finaliser is called. You are stuck
until the gc wakes up in the future to tidy it up.
Exception in thread "main" java.lang.Error at client.BufferedFileExample$MyBufferedReader.(BufferedFileExample.java:38) at client.BufferedFileExample.main(BufferedFileExample.java:12) Process exited with exit code 1.
Luckily there is an easy fix for this, the language change supports multiple entries in try statement, so you can modify this code to be:
// try (FileReader fr = new MyFileReader("/tmp/click.xml"); BufferedReader br = new MyBufferedReader( fr)) { System.out.println(br.readLine()); }
So if you run the code with the modification above then you find that close is now called:
Close called Exception in thread "main" java.lang.Error at client.BufferedFileExample$MyBufferedReader.(BufferedFileExample.java:38) at client.BufferedFileExample.main(BufferedFileExample.java:12) Process exited with exit code 1.
So problem solved, well unfortunately there is another wrinkle here, if we remove the code that throws the Error then we will see the following output:
Close called Close called Process exited with exit code 0.
So this is fine for classes that implement the existing java.io.Closeble interface because calling close for a second time is fine; but the more generic interface that is used for other closeable resources unfortunately doesn't have this restriction, so quote the spec
Note that unlike the close method of Closeable, this close method is not required to be idempotent. In other words, calling this close method more than once may have some visible side effect, unlike Closeable.close which is required to have no effect if called more than once. However, implementers of this interface are strongly encouraged to make their close methods idempotent.
So in order to using a split-try-with-resources idiom you first need to make sure that the resources have a idempotent close method. I suspect that most API will do this; but you need to check first and perhaps need to further wrap your objects to make this the case.
One final point with try-with-resource is the handling of exceptions when the close and construction fails. JDK 7 introduced the concept of suppressed exceptions, so you it make the split version of the code throw an exception both in the constructor or the MyBufferedReader and in the close of MyFileReader you get to see the following novel stack trace.
Close called Exception in thread "main" java.lang.Error at client.BufferedFileExample$MyBufferedReader.(BufferedFileExample.java:38) at client.BufferedFileExample.main(BufferedFileExample.java:12) Suppressed: java.lang.Error: close at client.BufferedFileExample$MyFileReader.close(BufferedFileExample.java:27) at client.BufferedFileExample.main(BufferedFileExample.java:14) Process exited with exit code 1.
This might not be what you calling code is expecting - certainly it might differ slightly from what the try/finally code you are replacing would produce, worth know thing that exceptions are handled slightly differently. For example if there is just an Error thrown during the close method would would see this odd self-referential stack trace.
Close called Close called Exception in thread "main" java.lang.Error: close at client.BufferedFileExample$MyFileReader.close(BufferedFileExample.java:28) at java.io.BufferedReader.close(BufferedReader.java:517) at client.BufferedFileExample.main(BufferedFileExample.java:15) Suppressed: java.lang.Error: close at client.BufferedFileExample$MyFileReader.close(BufferedFileExample.java:28) ... 1 more Process exited with exit code 1.
So the lesson here is that the impact of some of the new JDK 7 constructs are going to a little bit subtler than we might expect. Many thanks to my valued colleague Mark Warner for talking this one through with me.
Published at DZone with permission of Gerard Davison, DZone MVB. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments