java中try/catch性能和原理

来源:互联网 发布:新买ssd怎么装系统知乎 编辑:程序博客网 时间:2024/06/04 22:46

部分内容转载自http://blog.csdn.net/tao_zi7890/article/details/17584813

记得在做企业云项目的时候,我接了两个有意思的任务,一个是为几个线程加很多的try/catch代码。catch的异常有好几层,范围最小的,或者说最精准的异常放在最上面,最后跟一个Exception e。类似这种

try {} catch (IOException e) {} catch (Exception e) {}
还一个需求是在订单流转的三个线程里面增加监控的代码,监控这三个主要线程的执行时间,当时的工程师特意让我注意,增加的代码一定要写在try/catch里面,保证新增代码执行错误的时候,原有流程可以正常走下去。

正好今天有时间研究java中的try/catch执行原理,现在总结一下。

比如现在有两段代码

private void test1() {List<String> testList = new ArrayList<>();testList.add("1");testList.add("2");testList.add("a");testList.add("4");testList.add("5");for (String str : testList) {try {int num = Integer.parseInt(str);System.out.println(num);} catch(NumberFormatException ex) {System.out.println(ex);}}}
private void test2() {List<String> testList = new ArrayList<>();testList.add("6");testList.add("a");testList.add("8");testList.add("9");testList.add("10");try {for (String str : testList) {int num = Integer.parseInt(str);System.out.println(num);}} catch(NumberFormatException ex) {System.out.println(ex);}}

先说功能上的区别:test1()的异常捕获代码在循环里面,当程序出现异常后,剩下的正常数据依然可以跑完,而test2()异常捕获代码在循环外面,循环内出现异常整个处理就终止了。这样的话,一般情况下我们对异常数据的处理方式其实是正常数据跑通流程,异常数据抛出,test2()并不能满足这种需求。

下面来看看性能问题,之前有看到资料说,try/catch要写在循环外面,,防止出现性能的损失。不过今天看了try/catch的执行原理,结论是在没有抛出异常的情况下,性能完全没区别。

讨论帖地址:http://stackoverflow.com/questions/141560/should-try-catch-go-inside-or-outside-a-loop
异常处理帖地址:http://www.javaworld.com/javaworld/jw-01-1997/jw-01-hood.html?page=1
1、类会跟随一张 异常表(exception table),每一个try catch都会在这个表里添加行记录,每一个记录都有4个信息(try catch的开始地址,结束地址,异常的处理起始位,异常类名称)。
2、当代码在运行时抛出了异常时,首先拿着抛出位置到异常表中查找是否可以被catch(例如看位置是不是处于任何一栏中的开始和结束位置之间),如果可以则跑到异常处理的起始位置开始处理,如果没有找到则原地return,并且copy异常的引用给父调用方,接着看父调用的异常表。。。以此类推。

所以最终的结论是,要根据需求,选择合适的try/catch进行处理,满足功能。



0 0
原创粉丝点击