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()方法都被显式地调用。保证了资源最大程度的关闭。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值