关于try finally

package test;

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

public class Main {

    public static void main(String[] args) {
        System.out.println("function1: " + function1());
        System.out.println("function2: " + function2());
        System.out.println("function3: " + function3());
        System.out.println("function4: " + function4());
        function5();
        function6();
    }

    public static int function1() {
        try {
            return 30;
        } finally {
            return 60;
        }

    }

    public static int function2() {
        int i = 30;
        try {
            return i;
        } finally {
            i = 60;
        }

    }

    public static int function3() {
        List<Integer> list = new ArrayList();
        list.add(1);
        try {
            return list.get(0);
        } finally {
            list.set(0,  2);
        }
    }

    public static int function4(){
        try {
            System.out.println(1 / 0);
        } finally {
            return 60;
        }
    }

    public static void function5(){
        try {
            System.out.println(1 / 0);
        } finally {
            System.out.println("finally in func5");
        }
    }

    public static void function6(){
        try{
            System.exit(0);
        } finally{
            System.out.println("function6");
        }
    }
}

输出结果:

function1: 60
function2: 30
function3: 1
function4: 60
finally in func5
Exception in thread "main" java.lang.ArithmeticException: / by zero
    at test.Main.function5(Main.java:56)
    at test.Main.main(Main.java:13)

貌似可以得出结论:
1. 如果finally块中有return语句,会覆盖掉try中的return语句;
2. finally块中,改变try块中返回变量的值,如果该变量是基本类型,则finally块中改变变量值的语句不起作用;如果该变量是引用类型,则起作用;
3. finally块中的代码不一定执行,比如try中有System.exit(0);
4. try有导致抛出异常的语句,如果finally中有return语句,该异常不会抛出,否则会抛出。

《 The JavaTM Virtual Machine Specification, Second Edition 》有以下说明:
Java 虚拟机会把 finally 语句块作为 subroutine(子程序)直接插入到 try 语句块或者 catch 语句块的控制转移语句之前。但是,在执行 subroutine(也就是 finally 语句块)之前,try 或者 catch 语句块会保留其返回值到本地变量表(Local Variable Table)中。待 subroutine 执行完毕之后,再恢复保留的返回值到操作数栈中,然后通过 return 或者 throw 语句将其返回给该方法的调用者(invoker)。

先解释第2点” finally块中,改变try块中返回变量的值,如果该变量是基本类型,则finally块中改变变量值的语句不起作用;如果该变量是引用类型,则起作用”:
先看function2:

    public static int function2() {
        int i = 30;
        try {
            return i;
        } finally {
            i = 60;
        }
    }

“i = 60”被当成子程序插入到try中,但是在执行它之前,把30保存到了本地变量表,然后才执行i=60,但是又把30恢复到操作数栈,所以最后返回的还是30;
如果返回的是引用类型,保存到本地变量表的是地址,恢复的也是同一个地址,取值还是从内存中取,所以可以返回更新后的值。

再解释第4点” try有导致抛出异常的语句,如果finally中有return语句,该异常不会抛出,否则会抛出”:
看function4

    public static int function4(){
        try {
            System.out.println(1 / 0);
        } finally {
            return 60;
        }
    }

个人猜测,即使不写catch,编译后也会自带catch的东西,Java 虚拟机会把 finally 语句块作为 subroutine(子程序)直接插入到 try 语句块或者 catch 语句块的控制转移语句之前,然后return语句被插到抛出异常的语句前了。。。

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值