天天看点

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

https://blog.csdn.net/csdn_bang/article/details/86662998

1、finally是在return语句执行之后,return语句返回之前,finally是必执行的(当然是建立在try执行的基础上);

2、finally中修饰的基本类型没有return是不影响返回结果的,有了return才会影响返回结果;

3、finally中修改set、list、map引用类型时,就算没有return也会影响返回结果;

package com;

import java.util.ArrayList;
import java.util.List;

/**
 * @author wanjiadong
 * @date Create in 14:26 2019/2/13
 */
public class finallytest {

    public static void main(String[] args) {

        int j = test1();
        System.out.println("j=" +j);

        List<String> cats = new ArrayList<>();
        cats = addCats(cats);
        assert cats != null;
        cats.forEach(System.out::println);
    }

    private static int test1() {
        int i = 0;
        try {
            System.out.println("try");
            return i+=10;
        } catch (Exception e) {
            e.printStackTrace();
            return i;
        } finally {
            System.out.println("finally i =" + i);
            i+=10;
            System.out.println("finally ii="+i);
            return i;
        }
    }

    private static List<String> addCats(List<String> cats) {
        try {
            cats.add("a");
            return cats;
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            cats.add("b");
        }

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

    }
}
           

在JVM虚拟机种,有虚拟机栈,上面的代码中每一个方法都对应了一个栈帧,方法的执行对应的栈帧入栈,方法的执行完毕对应着栈帧的出栈。

栈帧可以理解为一个方法的运行空间。它主要由两部分构成,一部分是局部变量表,方法中定义的局部变量以及方法的参数就存放在这张表中;另一部分是操作数栈,用来存放操作数。

刚才的两段代码中的finally块中,i变量是要放到局部变量表的,每次有关于i的运算,都是要把i从局部变量表取出来(可以理解为copy一个副本),比如i += 10,那么需要把i和10都放到操作数栈中进行计算,然后得到一个结果,而这个结果是需要通过retrun语句写回到局部变量表。

第一段代码中的finally块中,虽然执行了i += 10,但是由于没有return,所以局部变量表中的内容没有变化,所以i还是10;

第二段代码中的finally块中,由于最后return i语句的执行,更新了局部变量中的i的值,所以最后返回的结果中i就是20了。

return返回后,就代表着方法执行结束,相应的该方法的栈帧就出栈了。而这个时候也就意味着,return返回是最后执行的,所以finally语句是在retrun返回之前执行的!