JDK7的try-with-resource方式的使用

如果使用传统try-catch-finally管理资源链接,代码可能是下面这样,finally代码远远多于业务代码。
这里写图片描述
为了增加代码可读性和可维护性,建议使用jdk7 提供的新特性try-with-resource(只能在表面上省去finally块关闭资源的逻辑):
这里写图片描述
即:try(资源定义){
业务逻辑
}
其实这只是个语法糖,因为编译时编译器会自动帮代码加上finally并调用close方法(前提是这些资源类都实现了Closeable接口)。(将你编译好的.class文件拖入idea即可看到编译后的代码(idea可以反编译出来))
这里写图片描述
可以看出finally代码块中除了正常关闭连接代码外,还包含了addSuppressed()方法,这个方法作用是保证一个异常不被另外一个异常抑制而无法抛出,比如try-catch块代码抛异常,程序会继续执行finally代码,但如果finally代码又抛错,就会导致try-catch的异常无法正常抛出,此时可以使用addSuppressed()方法可以将被抑制的异常也抛出。
参考链接:https://my.oschina.net/fhd/blog/324484
try-with-resource代码确实好用,程序可读性有所提高。但看着上面第二张图片的代码,你是否怀疑资源最大可能被关闭?是否编译器加上的fin.close()方法和out.close()方法就完整了?不是的。请看GZIPOutputStream类的close()方法
下面是GZIPOutputStream类的父类的close()方法(GZIPOutputStream类没重写)
这里写图片描述
close()方法时先调用finish()方法再调用out.close()方法,所以如果finish()方法顺利执行,那么out.close()方法确实可以顺利执行。
但进去finish()方法看看,
这里写图片描述
finish()方法是声明了会抛异常的,也就是finish()方法不见得一定正常执行,也就导致了out.close()方法在finish()方法抛异常时不能被调用,进而导致资源没有被关闭。
对于各个资源类的包装类,内部都是使用装饰者模式实现的,例子中调用out.close()方法,深层次还是调用FileOutputStream类的close方法,既然这样,我们程序就应该最大程度确保最内层资源的close()方法被调用(就算包装类的close()方法抛异常),才能最大程度上保证资源被关闭。
因此上面try-with-resource例子的流嵌套是不合理的,应该使用下面这种方式(分开定义每个Closeable类的资源):
这里写图片描述
看上面分开定义后编译生成的代码(如下):
这里写图片描述
显然,每一层Closeable类的资源的close()方法都被显式地调用。保证了资源最大程度的关闭。

发布了109 篇原创文章 · 获赞 130 · 访问量 44万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章