我們知道,在 Java 編程過程中固蛾,如果打開了外部資源(文件结执、數(shù)據(jù)庫連接、網(wǎng)絡(luò)連接等艾凯、redis)献幔,我們必須在這些外部資源使用完畢后,手動關(guān)閉它們趾诗。
因為外部資源不由 JVM 管理蜡感,無法享用 JVM 的垃圾回收機制,如果我們不在編程時確保在正確的時機關(guān)閉外部資源沧竟,就會導(dǎo)致外部資源泄露铸敏,緊接著就會出現(xiàn)文件被異常占用,數(shù)據(jù)庫連接過多導(dǎo)致連接池溢出**等諸多很嚴重的問題悟泵。
JDK7 之前的資源關(guān)閉方式
/**
* jdk7以前關(guān)閉流的方式
* */
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 語法關(guān)閉
在 JDK7 以前杈笔,Java 沒有自動關(guān)閉外部資源的語法特性,直到 JDK7 中新增了try-with-resource
語法糕非,才實現(xiàn)了這一功能蒙具。
try-with-resource Resource
的定義:所有實現(xiàn)了java.lang.AutoCloseable
接口(其中球榆,它包括實現(xiàn)了 java.io.Closeable 的所有對象),可以使用作為資源禁筏。
1. 一個例子
一個實現(xiàn)了 java.lang.AutoCloseable
接口的 Resource 類:
/**
* 資源類
* */
public class Resource implements AutoCloseable {
public void sayHello() {
System.out.println("hello");
}
@Override
public void close() throws Exception {
System.out.println("Resource is closed");
}
}
測試類 CloseResourceIn7.java
/**
* jdk7及以后關(guān)閉流的方式
* */
public class CloseResourceIn7 {
public static void main(String[] args) {
try(Resource resource = new Resource()) {
resource.sayHello();
} catch (Exception e) {
e.printStackTrace();
}
}
}
打印結(jié)果:
hello
Resource is closed
當(dāng)存在多個打開資源的時候: 資源二 Resource2.java
/**
* 資源2
* */
public class Resource2 implements AutoCloseable {
public void sayhello() {
System.out.println("Resource say hello");
}
@Override
public void close() throws Exception {
System.out.println("Resource2 is closed");
}
}
測試類 CloseResourceIn7.java
/**
* jdk7及以后關(guān)閉流的方式
* */
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();
}
}
}
結(jié)果:
hello
hello2
Resource2 is closed
Resource is closed
注意:打開多個資源的時候持钉,關(guān)閉順序是倒敘的
原理
查看編譯的 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();
}
}
}
可以發(fā)現(xiàn)編譯以后生成了 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");
}
@Override
public void close() throws Exception {
throw new Exception("Close method throw Exception");
}
}
測試類 CloseResourceBefore7.java
/**
* jdk7以前關(guān)閉流的方式
* */
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();
}
}
}
}
打印結(jié)果:
java.lang.Exception: Close method throw Exception
at com.shuwen.Resource.close(Resource.java:15)
at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:27)
at com.shuwen.CloseResourceIn7.main(CloseResourceIn7.java:12)
只打印了最后出現(xiàn)的關(guān)閉異常【異常屏蔽】這樣會給開發(fā)人員排查錯誤帶來一定的困難 我們換成 try-with-resource 方法實現(xiàn)CloseResourceIn7.java
/**
* jdk7及以后關(guān)閉流的方式
* */
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 Exception
at 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 Exception
at com.shuwen.Resource.close(Resource.java:15)
at com.shuwen.CloseResourceIn7.errorTest(CloseResourceIn7.java:21)
... 1 more
可以發(fā)現(xiàn)州刽,異常信息中多了一個 Suppressed 的提示空执,告訴我們這個異常其實由兩個異常組成,Close method throw Exception
這個異常是被 Suppressed【屏蔽】的異常穗椅。
應(yīng)用:在 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()) { // 用完自動 close
doSomething(jedis);
}
}
private static void doSomething(Jedis jedis) {
// code it here
}
}
這樣 Jedis 對象肯定會歸還給連接池 (死循環(huán)除外)辨绊,避免應(yīng)用程序卡死的慘劇發(fā)生。