测试案例准备:
创建一个AutoCloseableResource类实现了AutoCloseable()接口,重写close(调用时抛出异常)方法,自定义一个doExceptionSomething(调用时抛出异常)与一个doSomething(模拟正常逻辑执行)来进行测试。
测试不同的方法调用与异常捕捉情况。 复现异常抑制现象。
同时探究try-with-resources 反编译代码后的底层执行逻辑,Throwable.addSuppressed() / Throwable.getSuppressed()等,执行时机.....
先抛出我自己的6个问题,如下:
- try-with-resources语法糖会不会帮我们生成finally块?
- try块中不发生异常,try-with-resources自动调用close()方法时,出现了异常,catch是否能捕获close()方法中出现的异常(是否会被抑制)?
- try块中出现了异常后,try-with-resources自动调用close()方法,也出现一个异常,这个异常是否被抑制?
- getSuppressed如何来解决异常抑制情况?
- try-with-resources情况下,底层自动调用Suppressed数组中,存在所有的异常,还是只有被抑制的异常?
- 网上说多catch块能解决异常抑制,是否可行?
核心代码如下,部分测试案例有细微变动:
class AutoCloseableResource implements AutoCloseable {
public void doExceptionSomething() throws Exception {
throw new Exception("逻辑时处理出现异常");
}
public void doSomething() {
System.out.println("正常处理业务逻辑");
}
@Override
public void close() throws Exception {
throw new Exception("关闭时发生异常");
}
}
public class TryWithResourcesDemo {
public static void main(String[] args) {
try (AutoCloseableResource resource = new AutoCloseableResource()) {
resource.doExceptionSomething();
} catch (Exception e) {
System.out.println("捕获异常为---> Caught Exception: " + e.getMessage());
// Throwable[] suppressed = e.getSuppressed();
// System.out.println("Suppressed 数组的长度: " + suppressed.length);
// for (Throwable throwable : suppressed) {
// System.out.println("Suppressed Exception: " + throwable.getMessage());
// }
}
}
}
try-with-resources语法糖会不会帮我们生成finally块?
- 会 ,经过反编译查看生成了finally块
-
try块中不发生异常,try-with-resources自动调用close()方法时出现了异常,catch是否能捕获close()方法中出现的异常?
- 可以捕获
try块中出现了异常后,try自动调用close()方法,也出现一个异常,这个异常是否被抑制?
- 会被抑制。
这里只抛出了逻辑处理异常,但是自动调用close方法的异常被抑制了。
如何通过getSuppressed来解决异常抑制情况?
- 通过Throwable.getSuppressed()方法获取
传统的try-catch-finally需要手动的调用Throwable.addSuppressed,存入可能被抑制的异常。但通过try-with-resources反编译后看到,生成的代码会自动调用该方法存入,只需要Throwable[] suppressed = e.getSuppressed();获取到抑制数组打印即可
try-with-resources情况下,底层自动调用Suppressed数组中,存在所有的异常,还是只有被抑制的异常?
- 只有被抑制的异常
通过反编译后的代码可以看见,try-with-resources代码自动生成finally块中Suppressed数组存放异常,只有调用close方法时候才通过Throwable.addSuppressed存入 。
网上说多catch块能解决异常抑制,是否可行?
- 不行
底层调用的close方法中出现的异常,依然会被try块中的异常抑制。
其它内容以及总结为文档,如对文章内容有异议,欢迎各位大佬批评指正