看《Effective Java》第三版的时候,看到了其中建议将try-finally
替换为try-with-resources
。这个语法糖还算有意思,特此成文。java
Java库中有不少资源须要手动关闭,好比InputStream、OutputStream、java.sql.Connection等等。在此以前,一般是使用try-finally
的方式关闭资源;Java7以后,推出了try-with-resources
声明来替代以前的方式。 try-with-resources
声明要求其中定义的变量实现 AutoCloseable 接口,这样系统能够自动调用它们的close方法,从而替代了finally中关闭资源的功能。sql
举个栗子,下面是一个复制文件的方法,按照本来try-catch-finally
的写法:spa
// 一个简单的复制文件方法。
public static void copy(String src, String dst) {
InputStream in = null;
OutputStream out = null;
try {
in = new FileInputStream(src);
out = new FileOutputStream(dst);
byte[] buff = new byte[1024];
int n;
while ((n = in.read(buff)) >= 0) {
out.write(buff, 0, n);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (in != null) {
try {
in.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (out != null) {
try {
out.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
复制代码
能够看出,这种实现很是的丑陋。code
下面来看使用了try-with-resources
后的效果:接口
public static void copy(String src, String dst) {
try (InputStream in = new FileInputStream(src);
OutputStream out = new FileOutputStream(dst)) {
byte[] buff = new byte[1024];
int n;
while ((n = in.read(buff)) >= 0) {
out.write(buff, 0, n);
}
} catch (IOException e) {
e.printStackTrace();
}
}
复制代码
try-with-resources
将会自动关闭try()
中的资源,而且将先关闭后声明的资源。资源
若是不catch IOException就更加清爽了:it
public static void copy(String src, String dst) throws IOException {
try (InputStream in = new FileInputStream(src);
OutputStream out = new FileOutputStream(dst)) {
byte[] buff = new byte[1024];
int n;
while ((n = in.read(buff)) >= 0) {
out.write(buff, 0, n);
}
}
}
复制代码
那么try-with-resources
有什么神奇之处呢?到底作了什么呢?io
咱们先来看下AutoCloseable
接口:编译
public interface AutoCloseable {
void close() throws Exception;
}
复制代码
其中仅有一个close方法,实现AutoCloseable接口的类将在close方法中实现其关闭资源的功能。class
而try-with-resources
实际上是个语法糖,它将在编译时编译成关闭资源的代码。咱们将上述例子中的代码编译成class文件,再反编译回java文件,就能看到以下代码:
public static void copy(String var0, String var1) throws IOException {
FileInputStream var2 = new FileInputStream(var0);
Throwable var3 = null;
try {
FileOutputStream var4 = new FileOutputStream(var1);
Throwable var5 = null;
try {
byte[] var6 = new byte[1024];
int var7;
while((var7 = var2.read(var6)) >= 0) {
var4.write(var6, 0, var7);
}
} catch (Throwable var29) {
var5 = var29;
throw var29;
} finally {
if (var4 != null) {
if (var5 != null) {
try {
// 关闭FileOutputStream
var4.close();
} catch (Throwable var28) {
var5.addSuppressed(var28);
}
} else {
var4.close();
}
}
}
} catch (Throwable var31) {
var3 = var31;
throw var31;
} finally {
if (var2 != null) {
if (var3 != null) {
try {
// 关闭FileInputStream
var2.close();
} catch (Throwable var27) {
var3.addSuppressed(var27);
}
} else {
var2.close();
}
}
}
}
复制代码
除却处理异常相关的代码,其实就是调用了资源的close方法。
不过不得不说这个语法糖挺甜,真香。