`
ydbc
  • 浏览: 718728 次
  • 性别: Icon_minigender_1
  • 来自: 大连
文章分类
社区版块
存档分类
最新评论

提高你的Java代码质量吧:不要在finally块中处理返回值

 
阅读更多

一、分析

在finally中处理return返回值,代码上看上去很完美,都符合逻辑,但是执行起来就会产生逻辑错误,最重要的一点是finally是用来做异常处理的收尾处理的,一旦加上return语句,就会让程序的复杂程度陡然提升,而且会在一些隐蔽性非常高的错误。

与return语句相似,System.exit(0)或Runtime.getRuntime().exit(0)出现在异常代码块中也会产生非常多的错误假象

二、场景

如代码如下

public static void main(String[] args){ 
    try{ 
        doStuff(-1); 
        doStuff(100); 
    }catch(Exception e){ 
        System.out.println("这里是永远都不会到达的"); 
    } 
} 
 
//该方法抛出受检异常 
public static int doStuff(int _p)throws Exception{ 
    try{ 
       if(_p < 0){ 
           throw new DataFormatException("数据格式错误"); 
       }else{ 
           return _p; 
       } 
    }catch(Exception e){ 
        //异常处理 
        return e; 
    }finally{ 
        return -1; 
    } 
} 

doStuff(-1)的值是-1,doStuff(100)的值也是-1,调用doStuff方法永远都会抛出异常。原因就是我们在finally代码块中加入了return语句,而回导致这两个问题:

1.覆盖了try代码块中的return返回值

当执行doStuff(-1)时,doStuff方法产生了DataFormatException异常,catch块在捕捉此异常后直接抛出,之后代码块执行到finally代码块,就会重置返回值,结果就是-1了,也就出现了先返回,在执行finally,在重置返回值的情况。

2.屏蔽异常

为什么明明把异常throw出去了,但main方法捕捉不到呢?这是因为异常线程在监视到有异常发生时,就会登记当前的异常类型为DataFormatException,但是当执行器执行finally代码块时,则会重新为doStuff方法赋值,也就是告诉调用者“该方法执行正确,没有产生异常,返回值是1”,于是乎,异常就神奇的消失了。

三、建议

不要在finally代码块出现return语句


分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics