面试题:Java程序中 try,catch和finall 的执行顺序以及方法返回值

本文详细讨论了Java中try-catch-finally语句块中不同情况下的返回值行为。无论是否抛出异常,finally块总是会执行,并且其return值会覆盖try和catch中的返回值。总结了三种不同情况下方法返回值的确定规则,强调了finally在控制流程中的重要性及其对返回值的影响。
1. try,catch,finally中都有return值

执行下面的程序:

import java.util.*;
public static class Test1 {
    public static void main(String []args) {
        System.out.println("Output : "+test());
    }

    public static String test(){
        try{
            System.out.println("try");
            int i = new Random().nextInt(4);
            if (i>2) {
                System.out.println("throw");
                throw new Exception();
            }
            return "try";
        }catch(Exception e){
            System.out.println("catch");
            return "catch";
        }finally{
            System.out.println("finally");
            return "finally";
        }
    }
}

(1)如果异常发生时,输出:

try
throw
catch
finally
Output : finally

发生了异常时,先执行catch中的代码,再执行finally中的代码,但是最终方法返回结果还是finally中的return值

(2)如果没有异常发生时,输出:

try
finally
Output : finally

在没有异常发生时,先执行try中的代码,再执行finally中的代码,但是最终方法返回结果还是finally中的return值

所以当try,catch,finally中都有返回值时,不管有没有发生异常,由于finally总是最后执行,所以方法的返回值都是finally中的return值。

2. try,catch中有return值,finally中没有return值
public static class Test2 {
    public static void main(String []args) {
        System.out.println("Output : "+test());
    }

    public static String test(){
        try{
            System.out.println("try");
            int i = new Random().nextInt(4);
            if (i>2) {
                System.out.println("throw");
                throw new Exception();
            }
            return "try";
        }catch(Exception e){
            System.out.println("catch");
            return "catch";
        }finally{
            System.out.println("finally");
        }
    }
}

(1)如果异常发生时,输出:

try
throw
catch
finally
Output : catch

发生了异常时,先执行catch中的代码,再执行finally中的代码,最终方法返回结果还是catch中的return值。其实是先将catch中的返回结果暂存,执行完finally中的代码块之后再返回。

(2)如果没有异常发生时,输出:

try
finally
Output : try

在没有异常发生时,先执行try中的代码,再执行finally中的代码,但是最终方法返回结果还是try中的return值。其实是先将try中的返回结果暂存,执行完finally中的代码块之后再返回。

所以当try,catch有返回值,finally没有返回值时,不管有没有发生异常,finally总是最后执行。但是发生异常时方法的返回值都是catch中的return值,没有发生异常时方法的返回值时try中的return值。

此外还要注意,try,catch实际上是一个组合代码块,不能try中有返回值而catch中没有。如果catch中不加返回值,必须代码最后全局有一个返回值

3. try,catch中没有return值,finally中有return值
public static class Test3 {
    public static void main(String []args) {
        System.out.println("Output : "+test());
    }

    public static String test(){
        try{
            System.out.println("try");
            int i = new Random().nextInt(4);
            if (i>2) {
                System.out.println("throw");
                throw new Exception();
            }

        }catch(Exception e){
            System.out.println("catch");
        }finally{
            System.out.println("finally");
            return "finally";
        }
    }
}

(1)如果异常发生时,输出:

try
throw
catch
finally
Output : catch

发生了异常时,先执行catch中的代码,再执行finally中的代码,最终方法返回结果是finally中的return值。其实是这个时候finally中的返回值就是一个全局的返回值。

(2)如果没有异常发生时,输出:

try
finally
Output : try

在没有异常发生时,先执行try中的代码,再执行finally中的代码,最终方法返回结果是finally中的return值。其实是先将try中的返回结果暂存,执行完finally中的代码块之后再返回。

4. 结论
  • 不管有木有出现异常,finally块中代码都会执行;
  • 当try和catch中有return时,finally仍然会执行;
  • finally是在return后面的执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的;
  • finally中最好不要包含return,否则程序会提前退出,返回值不是try或catch中保存的返回值。但是根据需要可以自己判断要不要返回这个全局结果。

THE END.

### 使用 `try`、`catch`、`throw` `finally` 的方法 异常处理机制是编程中的重要组成部分,用于管理程序运行期间可能出现的错误情况。通过合理运用 `try`、`catch`、`throw` `finally` 关键字可以有效增强代码健壮性可维护性。 #### 基本结构说明 - **try**: 定义可能抛出异常的一段代码区域。 - **catch**: 处理由 `try` 中产生的特定类型的异常实例。 - **throw**: 显式地创建并抛出一个新的异常对象给调用者。 - **finally**: 不论是否发生异常都会被执行的部分;通常用来释放资源或执行清理工作[^1]。 下面是一个综合使用的例子: ```java public class ExceptionHandlingExample { public static void main(String[] args) { try { int result = divide(10, 0); System.out.println("Result is " + result); } catch (ArithmeticException e) { // Catch specific exceptions first. System.err.println("Caught ArithmeticException: " + e.getMessage()); throw new RuntimeException("Cannot proceed due to division by zero", e); // Rethrow with additional context. } finally { System.out.println("This will always execute."); } // Continue normal program flow after handling the exception... } private static int divide(int numerator, int denominator) throws ArithmeticException { if(denominator == 0){ throw new ArithmeticException("Denominator cannot be zero"); } return numerator / denominator; } } ``` 在这个案例里,当尝试除以零时会触发 `ArithmeticException` 异常,在对应的 `catch` 子句中捕获该异常并向外层重新抛出了带有更多信息的新异常。无论是否有异常被抛出,`finally` 部分都将得到执行,这有助于确保必要的收尾操作得以完成。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值