异常浅析之 try catch 嵌套和throw、throws

try catch 嵌套

try catch 肯定可以嵌套的,可以先来个嵌套小测试。

public static void main(String[] args) {
        try {
            System.out.println("**********************外层try**********************");
            errorMethod();
        } catch (Exception e) {
            System.out.println("**********************外层catch" + e + "**********************");
        } finally {
            System.out.println("**********************外层finally**********************");
        }
    }

    private static void errorMethod() {
        try {
            System.out.println("**********************内层try**********************");
            int i = 0;
            int a = 100 / i;
        } catch (Exception e) {
            System.out.println("**********************内层catch" + e + "**********************");
        } finally {
            System.out.println("**********************内层finally**********************");
        }
    }

内层函数体用零作被除数,报ArithmeticException。

运行结果是:

 若是注释掉内层catch,其他一样:

private static void errorMethod() {
        try {
            System.out.println("**********************内层try**********************");
            int i = 0;
            int a = 100 / i;
        } /*catch (Exception e) {
            System.out.println("**********************内层catch" + e + "**********************");
        }*/ finally {
            System.out.println("**********************内层finally**********************");
        }
    }

运行结果是:

 结论:

try catch 嵌套内层catch 可以捕获异常时,外层catch不会执行,但finally (多用于IO关闭)都会执行。

总结一个报错后的执行顺序:

    try {
            //A  报错地方
            try {
                //B 
            } catch (Exception e) {
                //C  
            } finally {
                //D  
            }
            //E  
        } catch (Exception e) {
            //F
        } finally {
            //G
        }

 执行顺序:

  • 内层A,E处抛出异常:由外层catch块捕获,并执行外层finally  ;
  • 内层B处抛出异常,且有一合适内层catch捕获在:执行内层finally,后执行E处  ;
  • 内层B处抛出异常,但内层catch块没有合适处理程序:执行内层finally,搜索外层catch,找合适的,执行外层finally,此时不会执行E  ;
  • 内层C处抛出异常在:退出内层catch块,执行内层finally,搜索外层catch,找到合适,执行外层finally  ;
  •  内层D处抛出异常在:退出内层finally块,搜索外层catch,找到合适,执行外层finally  ;

 

try catch 放循环体内还是循环外

网上很多争论这个,感兴趣读者可以看这里:https://hllvm-group.iteye.com/group/topic/27001

结论是:

语义不同的东西是不适合用来比较性能的。

但如果需求逻辑允许,把循环放在 try/catch 内会比把 try/catch 放在循环内要更有机会得到更好的优化。因为一般循环是重点优化对象,如果循环内部的控制流比较简单的话,分析和优化就会比较方便;把循环放在try/catch里面的话,就循环自己来看内部的控制流会比反过来放要简单一些。

 

throw、throws

抛出异常有三种形式,一是throw,一个throws,还有一种系统自动抛异常。

throw:  一般会用于程序出现某种逻辑时程序员主动抛出某种特定类型的异常。

throws:是方法可能抛出异常的声明。(用在声明方法时,表示该方法可能要抛出异常)

如:

if (httpResponse != null) {
     if (httpResponse.getStatusCode() == HttpStatus.SC_OK) {
                        Result result = JSON.parseObject(httpResponse.getResult(), Result.class);
           log.info("远程调用返回[result]={}", result);
     }else {
            log.warn("远程调用接口httpResponse返回异常码 : [url]={}, [httpResponse]={}", notifyUrl, httpResponse);
            throw new Exception("远程调用接口httpResponse返回异常码");
     }
}else {
     log.warn("远程调用接口返回空HttpResponse: [url]={} ", notifyUrl);
     throw new Exception("远程调用接口返回空HttpResponse");
}

这时候若是想在本方法内处理异常,则需要try catch 包住这段代码,若是不想处理,则抛出到上层处理,需在该方法后添加 throws Exception,如:

public void asyncDevStatusTes(List<String> deviceIdList,Integer opeStatus) throws Exception{
...
}

 

基础过后,说下我遇到的问题和解决办法:

@Async注解 与异常

在用异步线程池的时候,需要被调用函数向上层抛出异常,用 @Async注解后,调用函数捕获不到异常。而且 spring事务回滚也不起作用,都去掉异步注解后才能解决。原因是异步线程池会启用多个Task,这样事务里面的数据库操作 就不是一个任务了,事务不起作用。

 

至于书写异常可能会碰到的错误:Unhandled exception: Java.lang.exception

优质文章:try catch 的正确使用

参考文章:throw、throws

  • 5
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: try catch finally throw throws 是Java中的关键字,用于处理异常try:用于包含可能会抛出异常的代码块。 catch:用于捕获try块中抛出的异常,并进行相应的处理。 finally:无论try块中是否抛出异常,finally块中的代码都会被执行。 throw:用于手动抛出异常throws:用于声明方法可能会抛出的异常类型。 ### 回答2: trycatch、finally、throwthrows都是Java中用于处理异常的关键字。 trytry异常处理语句的开始,用于包裹可能会抛出异常的代码块。在try块中,可以放置可能会出现异常的代码逻辑。 catchcatch用于捕获try块中可能抛出的异常。如果try块中出现了异常,会检查catch块中是否有能够处理该异常的代码块。如果有匹配的catch块,程序会进入该catch块并执行异常处理逻辑。 finally:finally块用于定义无论是否发生异常,都必须执行的代码。即使try块中出现了异常,finally块中的代码仍然会被执行。finally块通常用于资源的释放,例如关闭文件或数据库连接。 throwthrow用于手动抛出异常。可以使用throw关键字在代码中显式地抛出某个具体的异常对象。一旦throw语句执行,程序将立即跳转到最近的catch块进行异常处理。 throws:throws关键字用于声明方法可能抛出的异常列表。在方法的声明中使用throws关键字,表明该方法可能会引发某些异常,调用该方法的代码需要对其进行异常处理或向上一层抛出。如果方法内的异常没有在catch块中处理,那么会根据throws所声明的异常列表传递给该方法的调用者。 总之,try用于尝试执行可能抛出异常的代码块,catch用于捕获和处理异常,finally用于定义无条件执行的代码块,throw用于手动抛出异常throws用于声明方法可能抛出的异常列表。这些关键字的使用可以有效地处理异常,提高程序的鲁棒性和可靠性。 ### 回答3: try catch finally throw throws 是Java中的异常处理机制。 try-catch语句用于捕获并处理异常try块中放置可能产生异常的代码,如果产生异常,则由catch块捕获并处理异常catch块中的代码是用于处理异常的,可以在其中编写特定的逻辑来处理异常情况。 finally块是可选的,用于在try-catch块之后执行一段无论是否发生异常都会执行的代码。finally块通常用于释放资源或进行清理操作,确保代码执行完毕后资源得到正确的释放。 throw关键字用于在代码中手动引发异常。通过throw语句,我们可以自定义异常并将其抛出,从而通知调用者发生了错误或特定情况。 throws关键字用于声明方法可能抛出的异常类型。在方法声明中使用throws关键字可以告诉调用者需要处理的可能出现的异常类型以便进行适当的异常处理。 总之,try-catch-finally语句用于处理可能发生的异常throw用于手动引发异常throws用于在方法声明中指定可能抛出的异常类型。异常处理是保证程序健壮性的重要一环,能够更好地应对异常情况,提高代码的可靠性和可维护性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值