美文网首页JAVA技术混杂
finally代码块中的代码一定会执行吗?

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

作者: Kinsanity | 来源:发表于2018-09-20 17:41 被阅读222次

    熟悉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代码块中的代码是一定会执行的代码了(哈哈)。

    相关文章

      网友评论

        本文标题:finally代码块中的代码一定会执行吗?

        本文链接:https://www.haomeiwen.com/subject/wbmnnftx.html