我们知道,在 Java 编程过程中,如果打开了外部资源(文件、数据库连接、网络连接等、redis),我们必须在这些外部资源使用完毕后,手动关闭它们。
因为外部资源不由 JVM 管理,无法享用 JVM 的垃圾回收机制,如果我们不在编程时确保在正确的时机关闭外部资源,就会导致外部资源泄露,紧接着就会出现文件被异常占用,数据库连接过多导致连接池溢出**等诸多很严重的问题。
JDK7 之前的资源关闭方式
/*** jdk7以前关闭流的方式* */public class CloseResourceBefore7 {private static final String FileName = "file.txt";public static void main(String[] args) throws IOException {FileInputStream inputStream = null;try {inputStream = new FileInputStream(FileName);char c1 = (char) inputStream.read();System.out.println("c1=" + c1);} catch (IOException e) {e.printStackTrace();} finally {if (inputStream != null) {inputStream.close();}}}}
JDK7 之后用 try-with-resource 语法关闭
在 JDK7 以前,Java 没有自动关闭外部资源的语法特性,直到 JDK7 中新增了try-with-resource 语法,才实现了这一功能。
try-with-resource Resource的定义:所有实现了java.lang.AutoCloseable接口(其中,它包括实现了 java.io.Closeable 的所有对象),可以使用作为资源。
一个例子
一个实现了 java.lang.AutoCloseable 接口的 Resource 类:
/*** 资源类* */public class Resource implements AutoCloseable {public void sayHello() {System.out.println("hello");}@Overridepublic void close() throws Exception {System.out.println("Resource is closed");}}
测试类 CloseResourceIn7.java
/*** jdk7及以后关闭流的方式* */public class CloseResourceIn7 {public static void main(String[] args) {try(Resource resource = new Resource()) {resource.sayHello();} catch (Exception e) {e.printStackTrace();}}}
打印结果:
helloResource is closed
当存在多个打开资源的时候: 资源二 Resource2.java
/*** 资源2* */public class Resource2 implements AutoCloseable {public void sayhello() {System.out.println("Resource say hello");}@Overridepublic void close() throws Exception {System.out.println("Resource2 is closed");}}
测试类 CloseResourceIn7.java
/*** jdk7及以后关闭流的方式* */public class CloseResourceIn7 {public static void main(String[] args) {try(Resource resource = new Resource(); Resource2 resource2 = new Resource2()) {resource.sayHello();resource2.sayhello();} catch (Exception e) {e.printStackTrace();}}}
结果:
hellohello2Resource2 is closedResource is closed
注意:打开多个资源的时候,关闭顺序是倒叙的
原理
查看编译的 class 文件 CloseResourceIn7.class
public class CloseResourceIn7 {public CloseResourceIn7() {}public static void main(String[] args) {try {Resource resource = new Resource();Throwable var2 = null;try {resource.sayHello();} catch (Throwable var12) {var2 = var12;throw var12;} finally {if (resource != null) {if (var2 != null) {try {resource.close();} catch (Throwable var11) {var2.addSuppressed(var11);}} else {resource.close();}}}} catch (Exception var14) {var14.printStackTrace();}}}
可以发现编译以后生成了 try-catch-finally 语句块 finally 中的 var2.addSuppressed(var11),这么做是为了处理异常屏蔽的。
我们将代码修改一下 资源 Resource.java,让两个方法里都抛出异常
/*** 资源类* */public class Resource implements AutoCloseable {public void sayHello() throws Exception {throw new Exception("Resource throw Exception");}@Overridepublic void close() throws Exception {throw new Exception("Close method throw Exception");}}
测试类 CloseResourceBefore7.java
/*** jdk7以前关闭流的方式* */public class CloseResourceBefore7 {public static void main(String[] args) {try {errorTest();} catch (Exception e) {e.printStackTrace();}}private static void errorTest() throws Exception {Resource resource = null;try {resource = new Resource();resource.sayHello();}finally {if (resource != null) {resource.close();}}}}
打印结果:
java.lang.Exception: Close method throw Exceptionat com.shuwen.Resource.close(Resource.java:15)at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:27)at com.shuwen.CloseResourceIn7.main(CloseResourceIn7.java:12)
只打印了最后出现的关闭异常【异常屏蔽】这样会给开发人员排查错误带来一定的困难 我们换成 try-with-resource 方法实现CloseResourceIn7.java
/*** jdk7及以后关闭流的方式* */public class CloseResourceIn7 {public static void main(String[] args) {try {errorTest();} catch (Exception e) {e.printStackTrace();}}private static void errorTest() throws Exception {try(Resource resource = new Resource()) {resource.sayHello();}}}
打印信息:
java.lang.Exception: Resource throw Exceptionat com.shuwen.Resource.sayHello(Resource.java:10)at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:20)at com.shuwen.CloseResourceIn7.main(CloseResourceIn7.java:12)Suppressed: java.lang.Exception: Close method throw Exceptionat com.shuwen.Resource.close(Resource.java:15)at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:21)... 1 more
可以发现,异常信息中多了一个 Suppressed 的提示,告诉我们这个异常其实由两个异常组成,Close method throw Exception这个异常是被 Suppressed【屏蔽】的异常。
应用:在 Jedis 中的使用
import redis.clients.jedis.Jedis;import redis.clients.jedis.JedisPool;public class JedisTest {public static void main(String[] args) {JedisPool pool = new JedisPool();try (Jedis jedis = pool.getResource()) { // 用完自动 closedoSomething(jedis);}}private static void doSomething(Jedis jedis) {// code it here}}
这样 Jedis 对象肯定会归还给连接池 (死循环除外),避免应用程序卡死的惨剧发生。
