zoukankan      html  css  js  c++  java
  • Try-Catch-Finally语句块执行问题

    Try-Catch-Finally语句块执行问题

    记录一个今天某公司的面试问题,其实我问题回答对了,但是面试官问我动手验证过没有,这还真没有,纯理论,被怼惨了,希望自己能变得更强大。

    Try-Catch-Finally语句块执行问题。

    一起来看下面这串代码:

    public class TryCatchFinally {
        public static void main(String[] args){
            System.out.println(get());
        }
        private static int get(){
            try{
                System.out.println("Try语句块");
                return 0;
            }catch (Exception e){
                System.out.println("Catch语句块");
                return 1;
            }finally {
                System.out.println("Finally语句块");
                return 2;
            }
        }
    }

    程序运行结果:

    再来看下面这串代码:

    public class TryCatchFinally {
        public static void main(String[] args){
            System.out.println(get());
        }
        private static int get(){
            try{
                System.out.println("Try语句块");
                throw new Exception();
            }catch (Exception e){
                System.out.println("Catch语句块");
                return 1;
            }finally {
                System.out.println("Finally语句块");
                return 2;
            }
        }
    }

    程序运行结果:

    总结:

    通过上面两个例子可以看出:

    • 无论是否在 try 语句块中抛出异常,finally语句块中的内容都会得到执行。
    • 只有 try 语句块中抛出异常了,catch语句块中的内容才会得到执行。
    • 但无论在 try 和 catch 语句块中是否有返回语句,finally 语句都会得到执行,并且当 finally 语句中有 return 语句,try 和 catch 语句中的 return 语句都无法得到执行。

    当然去掉 finally 中的 return 语句,try 或 catch 中的 return 语句又可以得到执行,这个可以直接在上面那个程序进行试验。

    同时 finally 语句块中包含 return 语句,编译器也会给出警告:finally block can not complete normally。

    这是因为 finally 的 return 语句覆盖了前面的 return 语句,是一种不合理的做法,尽量不要在 finally 中使用 return。

    补充一点:

    为什么 finally 语句始终都会得到执行,这里推荐一篇博客:https://blog.csdn.net/neosmith/article/details/48093427

    简单来说就是 JVM 将 finally 语句块中的东西都复制了一遍到 try 和 catch 语句块中,确保 finally语句块必定会得到执行。

    补充(2019/9/22)

    今天又发现一个问题,来更新一下,看下面这串代码:

    public class TestMain {
        public int test(){
            int a=0;
            try{
                a++;
                throw new Exception("故意的");
            }catch (Exception e){
                a++;
                return a;
            }finally {
                a++;
                System.out.println("a1="+a);
            }
        }
        public static void main(String[] args){
            int a=new TestMain().test();
            System.out.println("a2="+a);
        }
    }

    可以看到这个地方在test中打印的a值为 3 ,但返回值却是 2 ,这个地方又难住了,特地查了一下资料,有这么一句话:

    “Java 虚拟机会把 finally 语句块作为 subroutine直接插入到 try 语句块或者 catch 语句块的控制转移语句之前。但是,还有另外一个不可忽视的因素,那就是在执行 subroutine之前,try 或者 catch 语句块会保留其返回值到本地变量表中。待 subroutine 执行完毕之后,再恢复保留的返回值到操作数栈中,然后通过 return 或者 throw 语句将其返回给该方法的调用者。”

    也就是说这里是在 a++ 和 return 之间插入了 finally语句块,但是在执行 finally 语句块之前,先对 a 的值进行了保存,而之后的 finally语句块中对 a 进行的修改只是一个值传递,并没有对变量表中的 a 的值进行修改,所以也就是为什么在 finally 中 a 值为 3,而返回值 a 却为 2。

    当然这里如果将 a 改成一个对象,对对象中的某个值进行修改,也就是进行引用传递,则会对返回值进行修改。

    这里推荐一篇博客:https://www.jianshu.com/p/011062aaa855

     

     2019/9/22日更新

    吾生也有涯,而知也无涯。

  • 相关阅读:
    移动文件夹时,千万别直接移动系统文件夹,应该直接复制或者移动这个系统文件夹内的内容
    UIView的intrinsicContentSize方法,在按钮中重写
    Reveal v4(8796) 使用
    AppCode 2016.3 注册码
    BSD process name correspondlng to current thread: knernel_task Mac OS version Not yet set
    iOS 适配HTTPS
    [Graphics] UIColor created with component values far outside the expected range, Set a breakpoint on UIColorBreakForOutOfRangeColorComponents to debug. This message will only be logged once.
    SB中设置UITextField 无边框,真机上输入汉字聚焦时,文字 下沉
    TTTAttributedLabel 富文本小记
    iOS 应用数据存储的常用方式
  • 原文地址:https://www.cnblogs.com/hzauxx/p/11553208.html
Copyright © 2011-2022 走看看