try finally中的return问题

Java中try、finally语句中有return时的执行情况

博客分类: 

tryfinallyreturn执行顺序 

      在Java中当try、finally语句中包含return语句时,执行情况到底是怎样的,finally中的代码是否执行,大家众说纷纭,有的说会执行,有的说不会执行,到底哪种说法正确,现在通过下面的例子加以说明:

     第一种情况:try中包含return语句,finally中不包含

    

Java代码  

  1. public class TestTry {  
  2.     static String s="";  
  3.     public static void main(String args[]){  
  4.         s = test1();  
  5.         System.out.println("8  "+s);  
  6.     }  
  7.     public static String test1(){  
  8.           
  9.         try{  
  10.             System.out.println("try.....");  
  11.             return s = "a";  
  12.         }  
  13.         finally{  
  14.             s="b";    
  15.             System.out.println("17  "+s);  
  16.         }  
  17.     }  
  18. }  

  这里我们定义了一个字符串 s,在try里面将"a"赋值给s,并直接返回,在finally里面把"b"赋值给s,最终s的值是a还是b呢?下面是执行的结果

Java代码  

  1. try.....  
  2. 17  b  
  3. 8  a  

 我们发现最后的结果是a,但是b优先于a打印出来了,这是为什么呢?通过debug单步调试我们发现,在执行try中的return之前会先执行finally中的代码,之后再执行return语句。如果finally中也包含return语句,会出现什么情况呢,我们看第二种情况。

     第二种情况:try,finally中都包含return语句

      我们将上面的代码稍作改动

Java代码  

  1. public class TestTry {  
  2.     static String s="";  
  3.     public static void main(String args[]){  
  4.         s = test1();  
  5.         System.out.println("8  "+s);  
  6.     }  
  7.     public static String test1(){  
  8.           
  9.         try{  
  10.             System.out.println("try.....");  
  11.             return s = "a";  
  12.         }  
  13.         finally{  
  14.             return s="b";     
  15.               
  16.         }  
  17.     }  
  18. }  

 将finally中的s="b";直接改成returns="b";,会出现什么样的结果呢?字符串s是a还是b呢?

Java代码  

  1. try.....  
  2. 8  b  

 我们发现最后的打印结果是b。

 

结论:1.无论try里面有没有return语句,finally语句一定都会执行(不愧是finally,名字起的好)。

        2.如果finally中没有return语句,try里面有return,那么在执行try中的return语句之前会先去执行finally中的代码,再去执行try中的return语句;如果在finally中也包含return语句,将会直接返回,不再去执行try中的return语句。

 

关于java中的try-catch-finally语句和return

分类: Java程序设计基础2012-08-20 01:11 3348人阅读 评论(1) 收藏 举报

javaexceptionstringeclipseclass编译器

第一:return语句并不是函数的最终出口,如果有finally语句,这在return之后还会执行finally(return的值会暂存在栈里面,等待finally执行后再返回)

第二:finally里面不建议放return语句,根据需要,return语句可以放在try和catch里面和函数的最后。可行的做法有四:

1.return语句只在函数最后出现一次。

2.return语句仅在try和catch里面出现。

3.return语句仅在try和函数的最后都出现。

4.return语句仅在catch和函数的最后都出现。

注意,除此之外的其他做法都是不可行的,编译器会报错。

 

(1)如果程序运行到try成功时可以返回结果,则采用方法2。(见下面的例子test0_1,在那个例子中,方法2和4都是可行的,但是推荐方法2?)

(2)如果程序运行到catch时(即中途出错时)无需再继续执行后面的代码了,则采取方法4;(见下面例子中的test0,在那个特殊的例子中,只能采取方法4)

(3)如果程序运行到try或catch时还需要继续执行后面的代码,则采取方法1(见下面的例子test0_2,该例子只能采用方法1)。

 

下面是测试代码:

 

[java] viewplaincopyprint?

  1. public class Test {  
  2.   
  3.     public static void main(String[] args) {  
  4.         System.out.println("=============test1==================");  
  5.         System.out.println(test1());  
  6.         System.out.println("===============================");  
  7.   
  8.         System.out.println("=============test1_1==================");  
  9.         System.out.println(test1_1());  
  10.         System.out.println("===============================");  
  11.   
  12.         System.out.println("\n============test2===================");  
  13.         System.out.println(test2());  
  14.         System.out.println("===============================");  
  15.   
  16.         System.out.println("\n============test2_1===================");  
  17.         System.out.println(test2_1());  
  18.         System.out.println("===============================");  
  19.   
  20.         System.out.println("\n============test3===================");  
  21.         System.out.println(test3());  
  22.         System.out.println("===============================");  
  23.   
  24.         System.out.println("\n============test3_1===================");  
  25.         System.out.println(test3_1());  
  26.         System.out.println("===============================");  
  27.     }  
  28.   
  29.     public static String test0() {  
  30.         String a;  
  31.         int b;  
  32.         try{  
  33.             b = 8/0;  
  34.         }catch(Exception e){  
  35.             return null;  
  36.         }  
  37.         a = String.valueOf(b);  
  38.         return a+b;  
  39.     }  
  40.     
  41.     public static String test0_1() {  
  42.         String a;  
  43.         int b;  
  44.         try{  
  45.             b = 8/0;  
  46.             a = String.valueOf(b);  
  47.             return a+b;  
  48.         }catch(Exception e){  
  49.             return null;  
  50.         }  
  51.         //return a+b;  
  52.     }  
  53.       
  54.     public static String test0_2() {  
  55.         String a;  
  56.         int b=0;  
  57.         try{  
  58.             b = 8/0;  
  59.         }catch(Exception e){  
  60.         }  
  61.         a = String.valueOf(b);  
  62.         return a;  
  63.     }  
  64.       
  65.     public static String test1() {  
  66.         String a = "in try";  
  67.         int n = -1;  
  68.         try{  
  69.             return a+n; //先执行这个,再执行finally  
  70.         } catch ( Exception e ) {  
  71.   
  72.         } finally {  
  73.             //对String和int的更改均无效  
  74.             a = "in finally";  
  75.             n = 123;  
  76.             System.out.println("do finally");  
  77.         }  
  78.         return a; //不会执行  
  79.     }  //总结出一点:return语句并不是函数的最终出口,如果有finally语句,这在return之后还会执行finally  
  80.   
  81.     public static String test1_1() {  
  82.         String a = "in try";  
  83.   
  84.         try{  
  85.             return a;  
  86.         } catch ( Exception e ) {  
  87.   
  88.         } finally { //从eclpise报警告可看出,finally里面不建议有return语句  
  89.             a = "in finally";  
  90.             System.out.println("do finally");  
  91.             return a; //注释掉这句,eclipse将不再警告  
  92.         }  
  93.     }  
  94.   
  95.     public static int test2() {  
  96.         int a = 1;  
  97.   
  98.         try{  
  99.             return a;  
  100.         } catch ( Exception e ) {  
  101.   
  102.         } finally {  
  103.             a = 2;  
  104.             System.out.println("do finally");  
  105.         }  
  106.   
  107.         return a;  
  108.     } //很显然,finally里面更改无效,返回的是a=1  
  109.   
  110.     public static int test2_1() {  
  111.         int a = 1;  
  112.   
  113.         try{  
  114.             return a;  
  115.         } catch ( Exception e ) {  
  116.   
  117.         } finally {  
  118.             a = 2;  
  119.             System.out.println("do finally");  
  120.             return a;  
  121.         }  
  122.     } //很显然,a取finally里面的值,a=2  
  123.   
  124.     //Helper类,将整数转换成字符串  
  125.     static class Helper {  
  126.         int a;  
  127.   
  128.         public String toString() {  
  129.             return String.valueOf(a);  
  130.         }  
  131.     }  
  132.       
  133.     public static Helper test3() {  
  134.         Helper h = new Helper();  
  135.         h.a = 1;  
  136.   
  137.         try{  
  138.             return h;  
  139.         } catch ( Exception e ) {  
  140.   
  141.         } finally {  
  142.             h.a = 2; //对h.a的更改起作用!!  
  143.                     //因为在try里面返回的是一个句柄,它指向的对象的内容 是可以改变的  
  144.             System.out.println("do finally");  
  145.         }  
  146.           
  147.         return h; //这个不会被执行  
  148.     }  
  149.   
  150.     public static Helper test3_1() {  
  151.         Helper h = new Helper();  
  152.         h.a = 1;  
  153.   
  154.         try{  
  155.             return h;  
  156.         } catch ( Exception e ) {  
  157.   
  158.         } finally {  
  159.             h.a = 2; //返回a=2,这个不用说了  
  160.             System.out.println("do finally");  
  161.             return h;  
  162.         }  
  163.     }  
  164.   
  165.   
  166.     /** 
  167.      * 总结: 
  168.      * return语句,finally里面不建议放return语句,根据需要,可以放在try和catch里面 
  169.      *  
  170.      */  
  171.       
  172. }  

 

 异常处理之思考

    很少用自定义的异常类。

    因为对于一般的异常根本无需进行多么复杂的处理,所以用不着自定义的异常类)。如果有自定义的异常类,一般当场就catch掉而不抛出(抛出意味着什么?抛出一般会涉及到一个向上转型的过程,比如,为了能处理所有的异常,通常throws Exception,这样就会把所有的异常子类都向上转型为Exception父类,就失去了自定义异常类的意义)。

    但是实际上,对于异常,尽量做抛出处理,这样可以减少程序中到处都是try-catch。使用try-catch捕获异常时,建议一个try只放一行可能出现异常的代码,不要把很多无关的代码都写在try中(这是从编程的逻辑上而言,当多个异常集中在一起时,不妨只用一个try-catch。因为这样可以提高程序的效率,毕竟系统每处理一个try-catch语句都会有比较大的开销)

Java finally语句到底是在return之前还是之后执行?

网上有很多人探讨Java中异常捕获机制try...catch...finally块中的finally语句是不是一定会被执行?很多人都说不是,当然他们的回答是正确的,经过我试验,至少有两种情况下finally语句是不会被执行的:

(1)try语句没有被执行到,如在try语句之前就返回了,这样finally语句就不会执行,这也说明了finally语句被执行的必要而非充分条件是:相应的try语句一定被执行到。

(2)在try块中有System.exit(0);这样的语句,System.exit(0);是终止Java虚拟机JVM的,连JVM都停止了,所有都结束了,当然finally语句也不会被执行到。

 

当然还有很多人探讨Finally语句的执行与return的关系,颇为让人迷惑,不知道finally语句是在try的return之前执行还是之后执行?我也是一头雾水,我觉得他们的说法都不正确,我觉得应该是:finally语句是在try的return语句执行之后,return返回之前执行。这样的说法有点矛盾,也许是我表述不太清楚,下面我给出自己试验的一些结果和示例进行佐证,有什么问题欢迎大家提出来。

1. finally语句在return语句执行之后return返回之前执行的。

public class FinallyTest1 {

 

   public static void main(String[] args) {

       

       System.out.println(test1());

    }

 

   public static int test1() {

       int b = 20;

 

       try {

           System.out.println("try block");

 

           return b += 80;

       }

       catch (Exception e) {

 

           System.out.println("catch block");

       }

       finally {

           

           System.out.println("finally block");

           

           if (b > 25) {

               System.out.println("b>25, b = " + b);

           }

       }

       

       return b;

    }

   

}

运行结果是:

try block

finally block

b>25, b = 100

100

说明return语句已经执行了再去执行finally语句,不过并没有直接返回,而是等finally语句执行完了再返回结果。

如果觉得这个例子还不足以说明这个情况的话,下面再加个例子加强证明结论:

public class FinallyTest1 {

 

   public static void main(String[] args) {

       

       System.out.println(test11());

    }

   

   public static String test11() {

       try {

           System.out.println("try block");

 

          return test12();

     } finally {

          System.out.println("finally block");

      }

  }

 

 public static String test12() {

      System.out.println("return statement");

 

      return "after return";

   }

   

}

运行结果为:

try block

return statement

finally block

after return

说明try中的return语句先执行了但并没有立即返回,等到finally执行结束后再

这里大家可能会想:如果finally里也有return语句,那么是不是就直接返回了,try中的return就不能返回了?看下面。

2. finally块中的return语句会覆盖try块中的return返回。

public class FinallyTest2 {

 

   public static void main(String[] args) {

 

       System.out.println(test2());

    }

 

   public static int test2() {

       int b = 20;

 

       try {

           System.out.println("try block");

 

           return b += 80;

       } catch (Exception e) {

 

           System.out.println("catch block");

       } finally {

 

           System.out.println("finally block");

 

           if (b > 25) {

               System.out.println("b>25, b = " + b);

           }

 

           return 200;

       }

 

       // return b;

    }

 

}

运行结果是:

try block

finally block

b>25, b = 100

200

这说明finally里的return直接返回了,就不管try中是否还有返回语句,这里还有个小细节需要注意,finally里加上return过后,finally外面的return b就变成不可到达语句了,也就是永远不能被执行到,所以需要注释掉否则编译器报错。

这里大家可能又想:如果finally里没有return语句,但修改了b的值,那么try中return返回的是修改后的值还是原值?看下面。

3. 如果finally语句中没有return语句覆盖返回值,那么原来的返回值可能因为finally里的修改而改变也可能不变。

测试用例1:

public class FinallyTest3 {

 

   public static void main(String[] args) {

 

       System.out.println(test3());

    }

 

   public static int test3() {

       int b = 20;

 

       try {

           System.out.println("try block");

 

           return b += 80;

       } catch (Exception e) {

 

           System.out.println("catch block");

       } finally {

 

           System.out.println("finally block");

 

           if (b > 25) {

                System.out.println("b>25,b = " + b);

           }

 

           b = 150;

       }

 

       return 2000;

    }

 

}

运行结果是:

try block

finally block

b>25, b = 100

100

 

 

测试用例2:

 

import java.util.*;

 

public class FinallyTest6

{

   public static void main(String[] args) {

       System.out.println(getMap().get("KEY").toString());

    }

    

   public static Map<String, String> getMap() {

       Map<String, String> map = new HashMap<String, String>();

       map.put("KEY", "INIT");

         

       try {

           map.put("KEY", "TRY");

           return map;

       }

       catch (Exception e) {

           map.put("KEY", "CATCH");

       }

       finally {

           map.put("KEY", "FINALLY");

           map = null;

       }

        

       return map;

    }

}

运行结果是:

FINALLY

为什么测试用例1中finally里的b = 150;并没有起到作用而测试用例2中finally的map.put("KEY", "FINALLY");起了作用而map = null;却没起作用呢?这就是Java到底是传值还是传址的问题了,具体请看精选30道Java笔试题解答,里面有详细的解答,简单来说就是:Java中只有传值没有传址,这也是为什么map = null这句不起作用。这同时也说明了返回语句是try中的return语句而不是 finally外面的return b;这句,不相信的话可以试下,将return b;改为return 294,对原来的结果没有一点影响。

这里大家可能又要想:是不是每次返回的一定是try中的return语句呢?那么finally外的return b不是一点作用没吗?请看下面。

4. try块里的return语句在异常的情况下不会被执行,这样具体返回哪个看情况。

public class FinallyTest4 {

 

   public static void main(String[] args) {

 

       System.out.println(test4());

    }

 

   public static int test4() {

       int b = 20;

 

       try {

           System.out.println("try block");

 

           b = b / 0;

 

           return b += 80;

       } catch (Exception e) {

 

           b += 15;

           System.out.println("catch block");

       } finally {

 

           System.out.println("finally block");

 

           if (b > 25) {

               System.out.println("b>25, b = " + b);

           }

 

           b += 50;

       }

 

       return 204;

    }

 

}

运行结果是:

try block

catch block

finally block

b>25, b = 35

85

这里因为在return之前发生了除0异常,所以try中的return不会被执行到,而是接着执行捕获异常的catch 语句和最终的finally语句,此时两者对b的修改都影响了最终的返回值,这时return b;就起到作用了。当然如果你这里将return b改为return 300什么的,最后返回的就是300,这毋庸置疑。

这里大家可能又有疑问:如果catch中有return语句呢?当然只有在异常的情况下才有可能会执行,那么是在finally之前就返回吗?看下面。

5. 当发生异常后,catch中的return执行情况与未发生异常时try中return的执行情况完全一样。

public class FinallyTest5 {

 

   public static void main(String[] args) {

 

       System.out.println(test5());

    }

 

   public static int test5() {

       int b = 20;

 

       try {

           System.out.println("try block");

           

           b = b /0;

 

           return b += 80;

       } catch (Exception e) {

 

           System.out.println("catch block");

           return b += 15;

       } finally {

 

           System.out.println("finally block");

 

           if (b > 25) {

               System.out.println("b>25, b = " + b);

           }

 

           b += 50;

       }

 

       //return b;

    }

 

}

运行结果如下:

try block

catch block

finally block

b>25, b = 35

35

说明了发生异常后,catch中的return语句先执行,确定了返回值后再去执行finally块,执行完了catch再返回,finally里对b的改变对返回值无影响,原因同前面一样,也就是说情况与try中的return语句执行完全一样。

 

最后总结:finally块的语句在try或catch中的return语句执行之后返回之前执行且finally里的修改语句可能影响也可能不影响try或catch中 return已经确定的返回值,若finally里也有return语句则覆盖try或catch中的return语句直接返回。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值