finally的小特性

     try/catch/finally语句下,finally子句是肯定会执行的。但是很多人做不同的测试,却得出了不同的结论。

具体的原理最好是去看《深入java虚拟机》,里面对jsr、ret等几个指令做了详细的说明。这里不深入分析,而仅仅是从表现形式上看一下finally的特征。

代码:

  

Java代码   收藏代码
  1. /* 
  2.  * author: Zang XT 
  3.  */  
  4.   
  5. public class TestFinal {  
  6.     public static void main(String[] args) {  
  7.         System.out.println("test1:" + testFinal1());  
  8.         System.out.println("test2:" + testFinal2());  
  9.         System.out.println("test3:" + testFinal3());  
  10.         System.out.println("test4:" + testFinal4());  
  11.     }  
  12.   
  13.     static int testFinal1() {  
  14.         int i = 1;  
  15.         try {  
  16.             return i;  
  17.         } finally {  
  18.             System.out.println("in testFinal1():finally 肯定会被执行的!");  
  19.             i = 48;  
  20.         }  
  21.     }  
  22.   
  23.     static String testFinal2() {  
  24.         String str = "try";  
  25.         try {  
  26.             return str;  
  27.         } finally {  
  28.             System.out.println("in testFinal2():finally 肯定会被执行的!");  
  29.             str = "finally";  
  30.         }  
  31.     }  
  32.   
  33.     static StringBuilder testFinal3() {  
  34.         StringBuilder build = new StringBuilder("try ");  
  35.         try {  
  36.             return build;  
  37.         } finally {  
  38.             System.out.println("in testFinal3():finally 肯定会被执行的!");  
  39.             build.append("finally");  
  40.             build = new StringBuilder("你猜我是谁!");  
  41.         }  
  42.     }  
  43.   
  44.     static String testFinal4() {  
  45.         try {  
  46.             return "return in try";  
  47.         } finally {  
  48.             System.out.println("in testFinal4():finally 肯定会被执行的!");  
  49.             return "return in finally";  
  50.         }  
  51.     }  
  52. }  

 

 

输出是:

in testFinal1():finally 肯定会被执行的!

test1:1

in testFinal2():finally 肯定会被执行的!

test2:try

in testFinal3():finally 肯定会被执行的!

test3:try finally

in testFinal4():finally 肯定会被执行的!

test4:return in finally

     结论很明显,finally的语句确实执行了,而且肯定是在方法return之前执行的,而且,如果finally中有return语句的话,方法直接结束。这里需要注意的只有一点:在try中的return语句会将返回结果值压栈,然后转入到finally子过程,等到finally子过程执行完毕之后(没有return),再返回。
下面具体看4个例子:
      在testFinal1()中,return i;会将结果i的值,也就是1压入栈。即使在finally中将i修改了(i=48),也不回对已经压入栈里的1造成任何影响。
      在testFinal2()中,return str;将str的内容压入栈,比如我们假设str的内容为0x108(只是一个地址值),通过这个地址值我们能找到"try",那栈里的内容就是0x108。执行str = "finally",这时候str这个变量的内容可能变为0x237了,这是串"finally"的地址。方法调用结束后,返回的是什么?return时压入栈里的0x108。所以在打印结果时,我们打印的是通过0x108找到的字符串"try"。
      在testFinal3()中,return 压栈的是build这个变量的值,比如是0x3579,通过这个值我们可以找到StringBuilder对象。finally语句块中对这个对象的内容进行了修改。build = new StringBuilder("你猜我是谁!");让build变量指向了一个新的对象,这时候build的值可能是0x4579了。但是,别忘了,原来的StringBuilder对象仍然在0x3579处,而我们压栈的正是0x3579啊!方法返回后,我们得到的返回值0x3579,通过这个引用值找到相应的StringBuilder对象,所以打印的结果是test3:try finally。
      在testFinal4()中,finally有return语句,直接返回,方法结束。
      为什么不同的人有不同的结论?关键是没有正确理解压栈的是什么东西。其实初学java的时候,如果理解了变量是什么,并区分引用和对象本身就不会得到错误的结论了。再有,如果理解java中,方法调用都是采用传值模式的话,这里也就类似的可以明白了。

你可能感兴趣的:(finally的小特性)