JAVA技术混杂

finally代码块中的代码一定会执行吗?

2018-09-20  本文已影响222人  Kinsanity

熟悉java的人一定经常听说过,finally块中的代码一定会执行,但实际上真的是这样吗,本文带你看一下,java中有哪些情况finally中的代码不会执行。

先来解释下java中总是被人们放到一起比较的三个概念

final、finally、finalize的区别:

final: java中的修饰符。final修饰的类不能被继承,final修饰的方法不能被重写,final修饰的变量初始化之后不能被修改(当然这条不是绝对的,java中有一些手段可以修改)。
finally: java异常处理的组成部分,finally代码块中的代码一定会执行(如果真是这样本文也没必要存在了)。常用于释放资源。
finalize:Object类中的一个方法,垃圾收集器删除对象之前会调用这个对象的finalize方法。

finally与return:

public static int getNum(){
        int a = 10;
        try {
            a = 20;
            // a = a/0;
            return a;
        } catch (Exception e) {
            a = 30;
            return a;
        }finally{
            a = 40;
            //return a;
        }
    }//调用该方法返回20

有不少人认为该方法的返回值是40。执行完finally中的代码后a的值是40这是毋庸置疑的,但方法执行的结果为什么是20呢?那是因为在执行finally之前,程序将return结果赋值给临时变量,然后执行finally代码块,最后将临时变量返回。当然如果在finally代码块中有return语句,最终生效的是finally代码块中的return。

如果不明白请看下面的例子。

public static void main(String[] args) {
        System.out.println(getMap().get("name").toString());
        //打印zhaoliu
    }
    public static Map<String, String> getMap() {
        Map<String, String> map = new HashMap<String, String>();
        map.put("name", "zhangsan");     
        try {
            map.put("name", "lisi");
            return map;
        }
        catch (Exception e) {
            map.put("name", "wangwu");
        }
        finally {
            map.put("name", "zhaoliu");
            map = null;
        }         
        return map;
    }

有人认为上述代码在main方法中会报错,事实并非如此。以上代码更加印证了return返回的不是变量本身,而是这里的"临时变量".这里也说明了java中只有值传递没有引用传递。如果不明的的可以看我的另一篇文章【为什么大家都说java是值传递】

finally中的代码真的一定会执行吗?

今天给大家列举四种finally中的代码不会执行的情况(如果还有其它情况可以在评论里告知,我们共同学习)。

1.在执行异常处理代码之前程序已经返回

public static boolean getTrue(boolean flag) {
        if (flag) {
            return flag;
        }
        try {
            flag = true;
            return flag;
        } finally {
            System.out.println("我是一定会执行的代码?");
        }
    }

如果上述代码传入的参数为true那finally中的代码就不会执行了。

2.在执行异常处理代码之前程序抛出异常

public static boolean getTrue(boolean flag) {
        int i = 1/0;
        try {
            flag = true;
            return flag;
        } finally {
            System.out.println("我是一定会执行的代码?");
        }
    }

这里会抛出异常,finally中的代码同样不会执行。原理同1中差不多,只有与 finally 相对应的 try 语句块得到执行的情况下,finally 语句块才会执行。
就算try语句执行了finally中的代码一定会执行吗,答案是no,请看下面两种情况。

3.finally之前执行了System.exit()

public static boolean getTrue(boolean flag) {
        try {
            flag = true;
            System.exit(1);
            return flag;
        } finally {
            System.out.println("我是一定会执行的代码?");
        }
    }

System.exit是用于结束当前正在运行中的java虚拟机,参数为0代表程序正常退出,非0代表程序非正常退出。道理也很简单整个程序都结束了,拿什么来执行finally呢。

4.所有后台线程终止时,后台线程会突然终止

public static void main(String[] args) {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(5);
                } catch (Exception e) {
                }finally{
                    System.out.println("我是一定会执行的代码?");
                }
            }
        });
        t1.setDaemon(true);//设置t1为后台线程
        t1.start();
        System.out.println("我是主线程中的代码,主线程是非后台线程。");
    }

上述代码,后台线程t1中有finally块,但在执行前,主线程终止了,导致后台线程立即终止,故finally块无法执行(其实前面3点大家多多少少接触过,或是很容易理解,我主要想说的是这一点,一些人可能从来没听说过)。

总结:

1.与finally相对应的try语句得到执行的情况下,finally才有可能执行。
2.finally执行前,程序或线程终止,finally不会执行。
3.以后我再也不会说,fianlly代码块中的代码是一定会执行的代码了(哈哈)。
上一篇下一篇

猜你喜欢

热点阅读