想象这么一个情景,我们需要使用一个资源,在使用完之后需要关闭该资源,并且使用该资源的过程中有可能有异常抛出。此时我们都会想到用try-catch语句,在finally中关闭该资源。此时会有一个问题,如果关闭该资源的时候也抛出了异常呢?见如下例子:

  1. package com.rundong;
  2. import java.io.IOException;
  3. public class ResourceTest implements AutoCloseable{
  4. @Override
  5. public void close() throws IOException {
  6. throw new IOException("close Exception");
  7. }
  8. }
  1. package com.rundong;
  2. import java.io.IOException;
  3. import java.sql.SQLException;
  4. public class Main {
  5. public static void main(String[] args) {
  6. try {
  7. throwTest();
  8. } catch (SQLException e) {
  9. System.out.println("catch SQLException");
  10. e.printStackTrace();
  11. } catch (IOException e) {
  12. System.out.println("catch IOException");
  13. e.printStackTrace();
  14. }
  15. }
  16. static void throwTest() throws SQLException, IOException {
  17. ResourceTest resource = null;
  18. try {
  19. resource = new ResourceTest();
  20. throw new SQLException("the first Exception");
  21. } catch (SQLException e) {
  22. System.out.println("catch the first Exception");
  23. // do something
  24. throw e;
  25. } finally {
  26. if (resource != null) resource.close();
  27. }
  28. }
  29. }

输出结果:

 发现什么问题了吗?

finally中抛出的异常将try中抛出的异常覆盖了!我们无法捕获到第一个异常!

 

Java 7 的解决方式:

  1. package com.rundong;
  2. import java.io.IOException;
  3. import java.sql.SQLException;
  4. public class Main {
  5. public static void main(String[] args) {
  6. try {
  7. throwTest();
  8. } catch (SQLException e) {
  9. System.out.println("catch SQLException");
  10. e.printStackTrace();
  11. } catch (IOException e) {
  12. System.out.println("catch IOException");
  13. e.printStackTrace();
  14. }
  15. }
  16. static void throwTest() throws SQLException, IOException {
  17. try (ResourceTest resource = new ResourceTest()) {
  18. throw new SQLException("the first Exception");
  19. } catch (SQLException e) {
  20. System.out.println("catch the first Exception");
  21. // do something
  22. throw e;
  23. }
  24. }
  25. }

 

输出结果:

 try-with-resources的写法就能很好的处理这样的情景,它会自动关闭资源,并且如果关闭资源时也抛出了异常,不会覆盖原有异常的抛出。

使用条件:实现了AutoCloseable接口的类。

 

同时,这也提示着我们,永远不要尝试在finally中抛出异常!

版权声明:本文为liurundong原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/liurundong/p/15122273.html