前言
《Java 异常处理一览 | 基础篇》介绍了 Java 异常处理的一些基本操作,本文介绍下异常处理的一些进阶操作!
try-with-resources 语句
我们在对一些资源进行操作时,经常会有固定的写法:
- try 中打开资源
- finally 中关闭资源
比如下面这个程序:
static String readFirstLineFromFileWithFinallyBlock(String path)
throws IOException {
BufferedReader br = new BufferedReader(new FileReader(path));
try {
return br.readLine();
} finally {
br.close();
}
}
程序示例是从文件中读取第一行。它使用 BufferedReader
实例从文件中读取数据,这是一个必须在程序完成后关闭的资源。
在 Java SE 7 之前,对资源的操作只能通过上边比较繁琐的代码实现,也就是使用 finally
块确保关闭资源。
但 Java SE 7 之后,使用 try-with-resources
语句就可以做到无需在 finally
块中显式关闭资源,不管 try 语句是正常完成还是异常阻断,资源都会被自动关闭!写法如下:
static String readFirstLineFromFile(String path) throws IOException {
try (BufferedReader br =
new BufferedReader(new FileReader(path))) {
return br.readLine();
}
}
你会发现,try 语句多了一对圆括号,圆括号里能塞表达式,表达式里其实就是对资源的声明。
多资源处理
这时候你可能会问,如果 try 中我有多个资源要声明怎么办呢?
可以用分号分割。
像下面这样:
public static void writeToFileZipFileContents(String zipFileName,
String outputFileName)
throws java.io.IOException {
java.nio.charset.Charset charset =
java.nio.charset.StandardCharsets.US_ASCII;
java.nio.file.Path outputFilePath =
java.nio.file.Paths.get(outputFileName);
// Open zip file and create output file with
// try-with-resources statement
try (
java.util.zip.ZipFile zf =
new java.util.zip.ZipFile(zipFileName);
java.io.BufferedWriter writer =
java.nio.file.Files.newBufferedWriter(outputFilePath, charset)
) {
// Enumerate each entry
for (java.util.Enumeration entries =
zf.entries(); entries.hasMoreElements();) {
// Get the entry name and write it to the output file
String newLine = System.getProperty("line.separator");
String zipEntryName =
((java.util.zip.ZipEntry)entries.nextElement()).getName() +
newLine;
writer.write(zipEntryName, 0, zipEntryName.length());
}
}
}
进入 try
代码块的执行后,无论是正常还是异常,当代码块终止时,BufferedWriter
和 ZipFile
的 close
方法会依次被调用,从而达到自动关闭资源的效果。
注意:资源的 close
方法的调用顺序和它创建顺序相反,这个也好理解,你把 try
中涉及的所有资源想象依次放到一个比较窄的池子里,那这就像队列一样,先进后出,先创建的最后关闭资源。
原理解析
此时,你可能会问,为什么示例中的资源会被自动关闭资源呢?
最直接的原因就是,他们都有关闭资源的方法。
根本原因是,它们都实现了 AutoCloseable
的接口,具备了 close
的能力。
见名知意,AutoCloseable
也表明了实现它接口的对象,都具备自动关闭自己的能力。看注释,since 1.7,你也能知道,这个能力的确是从 Java 7 开始的。
所以,如果你有自定义资源,记得要实现 AutoCloseable
。