当前位置:   article > 正文

Java Catching and Handling Exceptions(二)

Java Catching and Handling Exceptions(二)

一、Try with resources语句

try with resources语句是声明一个或多个资源的try语句。资源是程序使用完后必须关闭的对象。try with resources语句确保在语句末尾关闭每个资源。任何实现java.lang.AutoCloseable的对象(包括实现java.io.Closeable的所有对象)都可以用作资源。
下面的示例从文件中读取第一行。它使用BufferedReader的实例从文件中读取数据。BufferedReader是一种资源,必须在程序完成后关闭:

  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-with-resources语句中声明的资源是BufferedReader。声明语句出现在try关键字后面的括号中。在Java SE 7和更高版本中,类BufferedReader实现了接口Java.lang.AutoCloseable。由于BufferedReader实例是在try with resource语句中声明的,因此无论try语句是正常完成还是突然完成(作为方法BufferedReader.readLine()引发IOException的结果),它都将被关闭。
在JavaSE7之前,您可以使用finally块来确保关闭资源,无论try语句是正常完成还是突然完成。下面的示例使用finally块而不是try with resources语句:

  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. }

然而,在本例中,如果方法readLine()和close都抛出异常,则方法readFirstLineFromFileWithFinallyBlock()抛出从finally块抛出的异常;从try块引发的异常被抑制。相反,在示例readFirstLineFromFile()中,如果从try块和try with resources语句中抛出异常,则方法readFirst LineFrom()抛出从try区块抛出的异常;从try-with-resources块引发的异常被抑制。在JavaSE7和更高版本中,您可以检索被抑制的异常;有关详细信息,请参阅抑制的异常一节。
您可以在try-with-resources语句中声明一个或多个资源。下面的示例检索压缩文件zipFileName中打包的文件的名称,并创建包含这些文件名称的文本文件:

  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-with-resources语句包含两个由分号分隔的声明:ZipFile和BufferedWriter。当直接跟在它后面的代码块正常或由于异常而终止时,BufferedWriter和ZipFile对象的close()方法将按此顺序自动调用。请注意,资源的close方法是按其创建的相反顺序调用的。
下面的示例使用try with resources语句自动关闭java.sql.statement对象:

  1. public static void viewTable(Connection con) throws SQLException {
  2. String query = "select COF_NAME, SUP_ID, PRICE, SALES, TOTAL from COFFEES";
  3. try (Statement stmt = con.createStatement()) {
  4. ResultSet rs = stmt.executeQuery(query);
  5. while (rs.next()) {
  6. String coffeeName = rs.getString("COF_NAME");
  7. int supplierID = rs.getInt("SUP_ID");
  8. float price = rs.getFloat("PRICE");
  9. int sales = rs.getInt("SALES");
  10. int total = rs.getInt("TOTAL");
  11. System.out.println(coffeeName + ", " + supplierID + ", " +
  12. price + ", " + sales + ", " + total);
  13. }
  14. } catch (SQLException e) {
  15. JDBCTutorialUtilities.printSQLException(e);
  16. }
  17. }

本例中使用的资源java.sql.Statement是JDBC 4.1和更高版本API的一部分。
注意:try-with-resources语句可以像普通的try语句一样具有catch和finally块。在try-with-resources语句中,任何catch或finally块都在声明的资源关闭后运行。

二、Suppressed Exceptions

可以从与try-with-resources语句关联的代码块中引发异常。在示例writeToFileZipFileContents()中,可以从try块引发异常,并且当try with resources语句尝试关闭ZipFile和BufferedWriter对象时,可以从该语句引发多达两个异常。如果从try块引发异常,并且从try with resources语句引发一个或多个异常,则从try with resources.语句引发的那些异常将被抑制,并且该块引发的异常是由writeToFileZipFileContents()方法引发的异常。通过从try块引发的异常中调用Throwable.getSuppressed()方法,可以检索这些被抑制的异常。

三、实现自动关闭或可关闭接口的类

请参阅AutoCloseable和Closeable接口的Javadoc,以获取实现这两个接口之一的类的列表。Closeable界面扩展了AutoCloseable接口。Closeable接口的close()方法抛出IOException类型的异常,而AutoCloseble接口的closer()方法则抛出Exception类的异常。因此,AutoCloseable接口的子类可以覆盖close()方法的这种行为,以抛出专门的异常,如IOException,或者根本没有异常。

四、将所有内容放在一起

前面的部分描述了如何在ListOfNumbers类中为writeList()方法构造try、catch和finally代码块。现在,让我们遍历代码并研究可能发生的情况。
将所有组件放在一起时,writeList()方法如下所示。

  1. public void writeList() {
  2. PrintWriter out = null;
  3. try {
  4. System.out.println("Entering" + " try statement");
  5. out = new PrintWriter(new FileWriter("OutFile.txt"));
  6. for (int i = 0; i < SIZE; i++) {
  7. out.println("Value at: " + i + " = " + list.get(i));
  8. }
  9. } catch (IndexOutOfBoundsException e) {
  10. System.err.println("Caught IndexOutOfBoundsException: "
  11. + e.getMessage());
  12. } catch (IOException e) {
  13. System.err.println("Caught IOException: " + e.getMessage());
  14. } finally {
  15. if (out != null) {
  16. System.out.println("Closing PrintWriter");
  17. out.close();
  18. }
  19. else {
  20. System.out.println("PrintWriter not open");
  21. }
  22. }
  23. }

如前所述,该方法的try块具有三种不同的退出可能性;这里有两个。

  1. try语句中的代码失败并引发异常。这可能是新FileWriter语句导致的IOException,也可能是for循环中错误的索引值导致的IndexOutOfBoundsException。
  2. 一切成功,try语句正常退出。

让我们看看在这两种退出可能性期间,writeList()方法中发生了什么。

1、场景1:发生异常

由于多种原因,创建FileWriter的语句可能会失败。例如,如果程序无法创建或写入所指示的文件,FileWriter的构造函数将引发IOException。
当FileWriter抛出IOException时,运行时系统立即停止执行try块;正在执行的方法调用未完成。然后,运行时系统开始在方法调用堆栈的顶部搜索适当的异常处理程序。在本例中,当IOException发生时,FileWriter构造函数位于调用堆栈的顶部。然而,FileWriter构造函数没有适当的异常处理程序,因此运行时系统检查方法调用堆栈中的下一个方法——writeList()方法。writeList()方法有两个异常处理程序:一个用于IOException,另一个用于IndexOutOfBoundsException。
运行时系统按照writeList()的处理程序在try语句之后出现的顺序来检查它们。第一个异常处理程序的参数是IndexOutOfBoundsException。这与引发的异常类型不匹配,因此运行时系统检查下一个异常处理程序-IOException。这与引发的异常类型匹配,因此运行时系统结束对适当异常处理程序的搜索。既然运行库已经找到了适当的处理程序,那么就执行该catch块中的代码。
在异常处理程序执行后,运行时系统将控制传递给finally块。finally块中的代码执行,而不管它上面捕获到什么异常。在这种情况下,FileWriter从未打开,也不需要关闭。在finally块完成执行后,程序继续执行finally模块之后的第一条语句。
这是ListOfNumbers程序的完整输出,在引发IOException时出现。 

  1. Entering try statement
  2. Caught IOException: OutFile.txt
  3. PrintWriter not open

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/44558
推荐阅读
相关标签
  

闽ICP备14008679号