try catch finally中return

今天记录一下try,catch,finally中的return。

先看一段代码:

下面代码的运行结果为?

public class test {
  public int add(int a,int b) {
    try {
      return a+b;
    }catch(Exception e){
      System.out.println("catch语句块");
    }finally {
      System.out.println("finally语句块");
    }
    return 0;
  }
  public static void main(String[] args) {
    test t=new test();
    System.out.println("和是"+t.add(9, 34));
  }
}

A、catch语句块 和是43

B、编译异常

C、finally语句块 和是43

D、和是43 finally语句块

正确答案:C

由于学习过编译原理的课程,知道了System.out.println中要执行add()方法与前面的“和是”字符串拼接后才会输出,因此首先执行add()方法。

add()方法中try语句块中有return语句,那么是否执行完try语句块就直接退出方法了呢?

上述代码在idea下运行此代码结果为:
try catch finally中return

 

 

 

看来尽管try语句块中有return,还是会执行finally语句块。

看到了合理的解释是:在try中执行到return语句时,不会真正的return,即只是会计算return中的表达式(本题为执行a+b),之后将结果保存在一个临时栈中,接着执行finally中的语句,最后才会从临时栈中取出之前的结果返回。

下面我们在这道题的finally语句中加入这么一行代码:

a=1;

 

public class test {
  public int add(int a,int b) {
    try {
      return a+b;
    }catch(Exception e){
      System.out.println("catch语句块");
    }finally {
      System.out.println("finally语句块");

      a=1;
    }
    return 0;
  }
  public static void main(String[] args) {
    test t=new test();
    System.out.println("和是"+t.add(9, 34));
  }
}

结果如何呢?

下面是运行结果截图:

try catch finally中return

 

 

从结果看出来结果并没有发生改变,这也验证了finally中的语句不会影响到临时栈中的值,即在执行finally之前,临时栈中的值已经确定为43了,执行finally语句将a的值变为1,对结果没有产生影响,执行完finally后的输出结果仍为43.

有了这些认识之后,我们讨论一下try,catch,finally中有return语句的几种情况。

第一种:try{}catch(){}finally{}return;

该情况语句后顺序执行。(不考虑异常)

第二种:try{return;}catch(){}finally{}return;

该情况为刚才说的题目情况,即执行完try语句块,将return的值保存在临时栈中,再执行finally语句块,之后返回临时栈中的值。

第三种:try{}catch(){return;}finally{}return;

无异常:执行try,执行finally,再执行return;

有异常:执行完catch语句块,将return的值保存在临时栈中,再执行finally语句块,之后返回临时栈中的值。

第四种:try{}catch(){}finally{return;}

执行finally中的return语句。

第五种:try{return;}catch(){return;}finally{};

根据有无异常执行和情况二或情况三。

第六种:try{return;}catch(){}finally{return;}

执行完try语句块,将return的值保存在临时栈中,再执行finally语句块,因为finally中有return,所以返回finally中的return值。

第七种:try{}catch(){return;}finally{return;}

执行完catch语句块,将return的值保存在临时栈中,再执行finally语句块,因为finally中有return,所以返回finally中的return值。

第八种:try{return;}catch(){return;}finally{return;}

有异常:执行情况七。

无异常:执行情况六。

注意:finally中最后不要有return语句,在编译器中会报警告。

总结:

在正常情况(即程序正常执行try catch finally语句块,不会在语句中出现退出程序、线程终止等特殊情况)下,都会执行finally语句块,如果finally中有return,则程序会走finally中的return,如果没有,则先执行try或者catch中的return,将其存入临时栈中,执行完finally语句后才返回临时栈中的值。

下面是看别人博客进行的总结,也在此分享一下:

 

对于含有return语句的情况,这里我们可以简单地总结如下:

  try语句在返回前,将其他所有的操作执行完,保留好要返回的值,而后转入执行finally中的语句,而后分为以下三种情况:

  情况一:如果finally中有return语句,则会将try中的return语句”覆盖“掉,直接执行finally中的return语句,得到返回值,这样便无法得到try之前保留好的返回值。

  情况二:如果finally中没有return语句,也没有改变要返回值,则执行完finally中的语句后,会接着执行try中的return语句,返回之前保留的值。

  情况三:如果finally中没有return语句,但是改变了要返回的值,这里有点类似与引用传递和值传递的区别,分以下两种情况,:

    1)如果return的数据是基本数据类型或文本字符串,则在finally中对该基本数据的改变不起作用,try中的return语句依然会返回进入finally块之前保留的值。

    2)如果return的数据是引用数据类型,而在finally中对该引用数据类型的属性值的改变起作用,try中的return语句返回的就是在finally中改变后的该属性的值。

  

 

上一篇:关于MYSQL 死锁的问题 Deadlock found when trying to get lock; try restarting


下一篇:捕获和抛出异常