'try(A a = new A())' VS 'try finally'

  • 2019 年 11 月 23 日
  • 筆記

实现了AutoCloseable接口的类,可以在try的时候直接实例化对象。try代码块完成之后,自动调用close方法,相当于在finally里主动调用。但是出现异常后的流程和try finally有什么不同呢? 下面写代码测试一下。

首先定义一个类Cat,实现AutoCloseable接口

class Cat implements AutoCloseable{      void sayHello() throws Exception {          Utils.println("calling sayHello(), I will throw an exception");          throw new Exception("Exception in sayHello() ");      }        @Override      public void close() throws Exception {          Utils.println("I'm closing, I will throw an exception");          throw new Exception("Exception in close()");      }  }

我们的这个Cat有以下特点:

  • sayHello方法会抛出异常
  • close方法也会抛出异常

test v1: 'try(Cat cat = new Cat())' VS 'try finally'

没有catch(不要被外层的catch迷惑,那只是为了打印异常)

static void testV1(){      Utils.println("----try(Cat cat = new Cat())-----");      try{          try(Cat cat = new Cat()){              cat.sayHello();          }        }catch (Exception e){          Utils.println("cache error in main (" + e + "), let's see its stack trace");          Utils.printStackTrace(e);      }      Utils.println("--------------");        Utils.println("----try finally-----");      try{          Cat cat = null;          try{              cat = new Cat();              cat.sayHello();          }finally {              if(cat != null){                  cat.close();              }          }      }catch (Exception e){          Utils.println("cache error in main (" + e + "), let's see its stack trace");          Utils.printStackTrace(e);      }      Utils.println("--------------");  }

结果输出:

----test v1----------------------------------------  ----try(Cat cat = new Cat())-----  calling sayHello(), I will throw an exception  I'm closing, I will throw an exception  cache error in main (java.lang.Exception: Exception in sayHello() ), let's see its stack trace  java.lang.Exception: Exception in sayHello()      at Cat.sayHello(Cat.java:4)      at Test.testV1(Test.java:16)      at Test.main(Test.java:4)      Suppressed: java.lang.Exception: Exception in close()          at Cat.close(Cat.java:10)          at Test.testV1(Test.java:17)          ... 1 more  --------------  ----try finally-----  calling sayHello(), I will throw an exception  I'm closing, I will throw an exception  cache error in main (java.lang.Exception: Exception in close()), let's see its stack trace  java.lang.Exception: Exception in close()      at Cat.close(Cat.java:10)      at Test.testV1(Test.java:33)      at Test.main(Test.java:4)  --------------

结论

  • try(Cat cat = new Cat())
    • try代码块完成之后会自动调用close
    • close抛出的异常,被Suppressed了,外层捕获的只有sayHello的异常,但通过堆栈可以找到这个Suppressed的异常
  • try finally
    • 外层捕获的是在finally执行close时抛出的异常,sayHello的异常完全不见了。

test v2: 'try(Cat cat = new Cat()) catch{}' VS 'try catch finally'

有catch,并且catch里再抛出异常

static void testV2(){      Utils.println("----try(Cat cat = new Cat()) catch-----");      try{          try(Cat cat = new Cat()){              cat.sayHello();          } catch (Exception e) {              Utils.println("cached err (" + e.getMessage() + "), I will throw an exception again");              throw new Exception("Exception in catch", e);          }        }catch (Exception e){          Utils.println("cache error in main (" + e + "), let's see its stack trace");          Utils.printStackTrace(e);      }      Utils.println("-----------------------------------------");        Utils.println("----try catch finally--------------------");      try{          Cat cat = null;          try{              cat = new Cat();              cat.sayHello();          } catch (Exception e) {              Utils.println("cached err (" + e.getMessage() + "), I will throw an exception again");              throw new Exception("Exception in catch", e);          }finally {              if(cat != null){                  cat.close();              }          }      }catch (Exception e){          Utils.println("cache error in main (" + e + "), let's see its stack trace");          Utils.printStackTrace(e);      }      Utils.println("-------------------------------------------");  }

结果输出

----test v2------  ----try(Cat cat = new Cat()){} catch{}-----  calling sayHello(), I will throw an exception  I'm closing, I will throw an exception  cached err (Exception in sayHello() ), I will throw an exception again  cache error in main (java.lang.Exception: Exception in catch), let's see its stack trace  java.lang.Exception: Exception in catch      at Test.testV2(Test.java:50)      at Test.main(Test.java:8)  -----------------------------------------  ----try catch finally--------------------  calling sayHello(), I will throw an exception  cached err (Exception in sayHello() ), I will throw an exception again  I'm closing, I will throw an exception  cache error in main (java.lang.Exception: Exception in close()), let's see its stack trace  java.lang.Exception: Exception in close()      at Cat.close(Cat.java:10)      at Test.testV2(Test.java:70)      at Test.main(Test.java:8)  -------------------------------------------  ---------------------------------------------------------------------

结论

  • try(Cat cat = new Cat()) catch
    • catch之前就调用了close(符合try代码块完成之后会自动调用close这个结论)
    • catch到的是sayHello的异常,close抛出的异常依然被Suppressed了
    • catch中再次抛出的异常被外层捕获
  • try catch finally
    • 先走catch,再走finally,所以catch捕获的是sayHello的异常
    • catch中再次抛出的异常不见了,外层捕获的是在finally执行close时抛出的异常。

测试代码地址:https://github.com/kongxiangxin/pine/tree/master/auto-closeable