java异常处理中的return和throw

如果把return和throw放在一起,直接会提示错误。"Unreachable statement"(无法被执行).

然而finally却可以成功骗过编译器让两者并存(是不是可以算是编译器的一个小bug呢),结果是后执行的会覆盖前者。

finally如果有return会覆盖catch里的throw,同样如果finally里有throw会覆盖catch里的return。

进而如果catch里和finally都有return finally中的return会覆盖catch中的。throw也是如此。

这样就好理解一些了,retrun和throw都是使程序跳出当前的方法,自然就是冲突的。如果非要跳出两次那么后者会覆盖前者。



public class T {

    public T() {

    }

    boolean testEx() throws Exception {

        boolean ret = true;

        try {

            ret = testEx1();

        } catch (Exception e) {

            System.out.println("testEx, catch exception");

            ret = false;

            throw e;

        } finally {

            System.out.println("testEx, finally; return value=" + ret);

            return ret;

        }

    }

    boolean testEx1() throws Exception {

        boolean ret = true;

        try {

            ret = testEx2();

            if (!ret) {

                return false;

            }

            System.out.println("testEx1, at the end of try");

            return ret;

        } catch (Exception e) {

            System.out.println("testEx1, catch exception");

            ret = false;

            throw e;

        }

        finally {

            System.out.println("testEx1, finally; return value=" + ret);

            return ret;

        }

    }

    boolean testEx2() throws Exception {

        boolean ret = true;

        try {

            int b = 12;

            int c;

            for (int i = 2; i >= -2; i--) {

                c = b / i;

                System.out.println("i=" + i);

            }

           return true;

        } catch (Exception e) {

            System.out.println("testEx2, catch exception");

            ret = false;

            throw e;

        }
        finally {

            System.out.println("testEx2, finally; return value=" + ret);

            //return ret;

        }
    }

    public static void main(String[] args) {

        T testException1 = new T();

        try {

            testException1.testEx();

        } catch (Exception e) {

            e.printStackTrace();

        }

    }

}
i=2
i=1
testEx2, catch exception
testEx2, finally; return value=false
testEx1, catch exception
testEx1, finally; return value=false
testEx, finally; return value=false

try-catch-finally程序块的执行流程

首先执行的是try语句块中的语句,这时可能会有以下三种情况:

1.              如果try块中所有语句正常执行完毕,那么finally块的居于就会被执行,这时分为以下两种情况:

        如果finally块执行顺利,那么整个try-catch-finally程序块正常完成。

        如果finally块由于原因R突然中止,那么try-catch-finally程序块的结局是“由于原因R突然中止(completes abruptly)”

2.              如果try语句块在执行过程中碰到异常V,这时又分为两种情况进行处理:

        如果异常V能够被与try相应的catch块catch到,那么第一个catch到这个异常的catch块(也是离try最近的一个与异常V匹配的catch块)将被执行;这时就会有两种执行结果:

        如果catch块执行正常,那么finally块将会被执行,这时分为两种情况:

        如果finally块执行顺利,那么整个try-catch-finally程序块正常完成。

        如果finally块由于原因R突然中止,那么try-catch-finally程序块的结局是“由于原因R突然中止(completes abruptly)”

        如果catch块由于原因R突然中止,那么finally模块将被执行,分为两种情况:

        如果如果finally块执行顺利,那么整个try-catch-finally程序块的结局是“由于原因R突然中止(completes abruptly)”。

        如果finally块由于原因S突然中止,那么整个try-catch-finally程序块的结局是“由于原因S突然中止(completes abruptly)”,原因R将被抛弃。

虽然我们在testEx2中使用throw e抛出了异常,但是由于testEx2中有finally块,而finally块的执行结果是complete abruptly的。因为return也是一种导致complete abruptly的原因之一,所以整个try-catch-finally程序块的结果是“complete abruptly”,所以在testEx1中调用testEx2时是捕捉不到testEx1中抛出的那个异常的,而只能将finally中的return 结果获取到。当然这种情况是可以避免的,以testEx2为例:如果你一定要使用finally而且又要将catch中 throw的e在testEx1中被捕获到,那么你去掉testEx2中的finally中的return就可以了。如果将testEx2()中的

finally {

            System.out.println("testEx2, finally; return value=" + ret);

            //return ret;

        }
修改为

finally {

            System.out.println("testEx2, finally; return value=" + ret);

            return ret;

        }
那么运行结果:

i=2
i=1
testEx2, catch exception
testEx2, finally; return value=false
testEx1, finally; return value=false
testEx, finally; return value=false

try-catch-finally程序块中的return

从上面的try-catch-finally程序块的执行流程以及执行结果一节中可以看出无论try或catch中发生了什么情况,finally都是会被执行的,那么写在try或者catch中的return语句也就不会真正的从该函数中跳出了,它的作用在这种情况下就变成了将控制权(语句流程)转到 finally块中;这种情况下一定要注意返回值的处理。

例如,在try或者catch中return false了,而在finally中又return true,那么这种情况下不要期待你的try或者catch中的return false的返回值false被上级调用函数获取到,上级调用函数能够获取到的只是finally中的返回值,因为try或者catch中的return 语句只是转移控制权的作用。

  • 9
    点赞
  • 35
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
throw 关键字在 Java 异常处理用于手动抛出一个异常对象。对于学生总成绩的计算,如果计算结果超出了合法的范围(例如负数或超过满分),我们可以通过 throw 抛出一个自定义的异常对象。 例如,我们可以定义一个自定义的异常类 TotalScoreException,表示学生总成绩异常。当计算出的总成绩不合法时,我们可以创建一个 TotalScoreException 对象,并使用 throw 关键字将其抛出,让调用该方法的上层代码进行异常处理。 示例代码如下: ``` public class TotalScoreException extends Exception { public TotalScoreException(String message) { super(message); } } public class Student { private int score1; private int score2; public Student(int score1, int score2) { this.score1 = score1; this.score2 = score2; } public int getTotalScore() throws TotalScoreException { int totalScore = score1 + score2; if (totalScore < 0 || totalScore > 100) { throw new TotalScoreException("Invalid total score: " + totalScore); } return totalScore; } } public class Main { public static void main(String[] args) { Student student = new Student(80, 90); try { int totalScore = student.getTotalScore(); System.out.println("Total score: " + totalScore); } catch (TotalScoreException e) { System.out.println("Error: " + e.getMessage()); } } } ``` 在上面的代码,当计算出的总成绩不合法时,我们使用 throw 抛出一个 TotalScoreException 对象,并将其包含的错误信息输出到控制台。在调用 getTotalScore() 方法时,我们使用 try-catch 块捕获可能抛出的异常,并在 catch 块处理异常情况。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值