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日更新

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

Try-Catch-Finally语句块执行问题的更多相关文章

  1. try...catch...finally语句块

    try-catch-finally语句主要是用来处理检查异常,捕获并处理,以及最后必须要执行的finally块. try-catch-finally语句入门: 1.try-catch-finally语 ...

  2. try catch finally语句块中存在return语句时的执行情况剖析

    2种场景 (1) try中有return,finally中没有return(注意会改变返回值的情形);(2) try中有return,finally中有return; 场景代码分析(idea亲测) 场 ...

  3. 顽强的的砂锅之——深究finally代码块与return语句的执行顺序!

    当问到finally代码块的执行顺序,就算刚刚学编程的小白都能毫不犹豫的说出答案:不管异常发生与否,finally语句块的代码一定会被执行!大体上这样讲是没有错,但是finally块中的代码一定会有效 ...

  4. Java中try catch finally语句中含return语句的执行情况总结-编程陷阱

    前言:有java编程基础的人对java的异常处理机制都会有一定了解,而且可能感觉使用起来也比较简单,但如果在try catch finally语句块中遇到return语句,开发者可能就会遇到一些逻辑问 ...

  5. Java静态语句块、语句块、构造方法执行顺序

    package com.imooc.practice; class Parent{ public Parent(){ System.out.println("Parent构造方法执行!&qu ...

  6. static{ }语句块详解

    static{}(即static块),会在类被加载的时候执行且仅会被执行一次,一般用来初始化静态变量和调用静态方法.举ge例子: public class Test { public static i ...

  7. 转 java中static{}语句块详解

    原文地址:http://blog.csdn.net/lubiaopan/article/details/4802430     感谢原作者! static{}(即static块),会在类被加载的时候执 ...

  8. java中static{}语句块详解

    static{}(即static块),会在类被加载的时候执行且仅会被执行一次,一般用来初始化静态变量和调用静态方法,下面我们详细的讨论一下该语句块的特性及应用. 一.在程序的一次执行过程中,stati ...

  9. 关于java的static语句块

    声明:转载请注明出处 static{}(即static块),会在类被加载的时候执行且仅会被执行一次,一般用来初始化静态变量和调用静态方法,下面我们详细的讨论一下该语句块的特性及应用. 一.在程序的一次 ...

随机推荐

  1. Mysql8.0主从复制搭建,shardingsphere+springboot+mybatis读写分离

    1.安装mysql8.0 首先需要在192.167.3.171上安装JDK. 下载mysql安装包,https://dev.mysql.com/downloads/,找到以下页面下载. 下载后放到li ...

  2. 7-SQL-join连接

    (1) 内连接 关键字:inner join on select * from a_table inner join b_table on a_table.a_id = b_table.b_id; / ...

  3. Android刷机

    1.安装第三方recovery 下载自己手机适配的recovery包 https://twrp.me/lg/lgnexus5.html fastboot 卡在 waiting for device $ ...

  4. 04webpack--webpack-dev-server 时时跟新

    <!-- 如何添加npm run dev 启动程序 下载npm i webpack-dev-server -S 在pack.json中添加 "dev": "webp ...

  5. 第50 课C++对象模型分析——成员函数(上)

    类中的成员函数位于代码段中调用成员函数时对象地址作为参数隐式传递成员函数通过对象地址访问成员变量C++语法规则隐藏了对象地址的传递过程 #include<iostream> #includ ...

  6. C++ class内的 < 和 > 重载,大于号,小于号,重载示例。

    #include <iostream> // overloading "operator = " outside class // < 和 > 是二元操作符 ...

  7. golang 服务端和客户端(二)

    1.golang服务端 package main import( "net/http" ) func main(){ //注册处理函数,用户连接,自动调用指定的处理函数 http. ...

  8. 日志检索实战 grep sed

    日志检索实战 grep sed 参考 sed命令 使用 grep -5 'parttern' inputfile //打印匹配行的前后5行 grep -C 5 'parttern' inputfile ...

  9. imutils.path

    from imutils import paths # 要在哪条路径下查找 path = '...' # 查找图片,得到图片路径 imagePaths = list(imutils.paths.lis ...

  10. Python str & repr

    Python str & repr repr 更多是用来配合 eval 的 (<- 点击查看),str 更多是用来转换成字符串格式的 str() & repr() str() 和 ...