前言

《Java 异常处理一览 | 基础篇》介绍了 Java 异常处理的一些基本操作,本文介绍下异常处理的一些进阶操作!

try-with-resources 语句

我们在对一些资源进行操作时,经常会有固定的写法:

  • try 中打开资源
  • finally 中关闭资源

比如下面这个程序:

  1. static String readFirstLineFromFileWithFinallyBlock(String path)
  2. throws IOException {
  3. BufferedReader br = new BufferedReader(new FileReader(path));
  4. try {
  5. return br.readLine();
  6. } finally {
  7. br.close();
  8. }
  9. }

程序示例是从文件中读取第一行。它使用 BufferedReader 实例从文件中读取数据,这是一个必须在程序完成后关闭的资源。

在 Java SE 7 之前,对资源的操作只能通过上边比较繁琐的代码实现,也就是使用 finally 块确保关闭资源。

但 Java SE 7 之后,使用 try-with-resources 语句就可以做到无需在 finally 块中显式关闭资源,不管 try 语句是正常完成还是异常阻断,资源都会被自动关闭!写法如下:

  1. static String readFirstLineFromFile(String path) throws IOException {
  2. try (BufferedReader br =
  3. new BufferedReader(new FileReader(path))) {
  4. return br.readLine();
  5. }
  6. }

你会发现,try 语句多了一对圆括号,圆括号里能塞表达式,表达式里其实就是对资源的声明

多资源处理

这时候你可能会问,如果 try 中我有多个资源要声明怎么办呢?

可以用分号分割。

像下面这样:

  1. public static void writeToFileZipFileContents(String zipFileName,
  2. String outputFileName)
  3. throws java.io.IOException {
  4. java.nio.charset.Charset charset =
  5. java.nio.charset.StandardCharsets.US_ASCII;
  6. java.nio.file.Path outputFilePath =
  7. java.nio.file.Paths.get(outputFileName);
  8. // Open zip file and create output file with
  9. // try-with-resources statement
  10. try (
  11. java.util.zip.ZipFile zf =
  12. new java.util.zip.ZipFile(zipFileName);
  13. java.io.BufferedWriter writer =
  14. java.nio.file.Files.newBufferedWriter(outputFilePath, charset)
  15. ) {
  16. // Enumerate each entry
  17. for (java.util.Enumeration entries =
  18. zf.entries(); entries.hasMoreElements();) {
  19. // Get the entry name and write it to the output file
  20. String newLine = System.getProperty("line.separator");
  21. String zipEntryName =
  22. ((java.util.zip.ZipEntry)entries.nextElement()).getName() +
  23. newLine;
  24. writer.write(zipEntryName, 0, zipEntryName.length());
  25. }
  26. }
  27. }

进入 try 代码块的执行后,无论是正常还是异常,当代码块终止时,BufferedWriterZipFileclose 方法会依次被调用,从而达到自动关闭资源的效果。

注意:资源的 close 方法的调用顺序和它创建顺序相反,这个也好理解,你把 try 中涉及的所有资源想象依次放到一个比较窄的池子里,那这就像队列一样,先进后出,先创建的最后关闭资源。

原理解析

此时,你可能会问,为什么示例中的资源会被自动关闭资源呢?

最直接的原因就是,他们都有关闭资源的方法

根本原因是,它们都实现了 AutoCloseable 的接口,具备了 close 的能力。

image.png

见名知意,AutoCloseable 也表明了实现它接口的对象,都具备自动关闭自己的能力。看注释,since 1.7,你也能知道,这个能力的确是从 Java 7 开始的。

所以,如果你有自定义资源,记得要实现 AutoCloseable