`
陈亮宇
  • 浏览: 12279 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
文章分类
社区版块
存档分类
最新评论
阅读更多

浅谈项目中异常处理
  一个好的项目好的异常处理机制是不可或缺的,而作为程序员对异常的处理方法的优劣对项目的调试和性能往往也会产生意想不到的效果 “本文是Exception处理的一篇不错的文章,从Java Exception的概念介绍起,依次讲解了Exception的类型(Checked/Unchecked),Exception处理的最佳实现:      1. 选择Checked还是Unchecked的几个经典依据      2. Exception的封装问题      3. 如无必要不要创建自己得Exception      4. 不要用Exception来作流程控制      5. 不要轻易的忽略捕获的Exception      6. 不要简单地捕获顶层的Exception”      ——选自JAVADigest.Net对原文的介绍                “JAVADigest.Net这个站点不知道大家是否经常上,就像它的名字一样,它让我们更加有效的消化Java,或者它就像个中转站一样,至少对我是这样的,有些好的可以说是非常经典的技术文章,我都是通过它第一次获得,更多的时候我是为了偷懒才上JAVADigest.Net,因为如果是近期比较经典的文章,它上边都有介绍文字和原文连接。”       ——小插曲并非常荣幸地推荐JAVADigest.Net给你                                  关于异常处理的一个问题就是要对何时(when)和如何(how)使用它们做到了然于心。在本文中我将介绍一些关于异常处理的最佳实践,同时我也会涉及到最近争论十分激烈的checked Exception的使用问题。      作为开发员,我们都希望能写出解决问题并且是高质量的代码。不幸的是,一些副作用(side effects)伴随着异常在我们的代码中慢慢滋生。无庸置疑,没有人喜欢副作用(side effects),所以我们很快就用我们自己的方式来避免它,我曾经看到一些聪明的程序员用下面的方式来处理异常:               public void consumeAndForgetAllExceptions(){    try {    ...some code that throws exceptions    } catch (Exception ex){    ex.printStacktrace();    }   }                  上边的代码有什么问题么?               在回答以前让我们想想怎样才是正确的?是的,一旦程序碰到异常,它就该挂起程序而“做”点什么。那么上边的代码是这样子的么?看吧,它隐瞒了什么?它把所有的“苦水”往肚里咽(在控制台打印出异常信息),然后一切继续,从表面上看就像什么都没有发生过一样……,很显然,上边代码达到的效果并不是我们所期望的。               后来又怎样?               public void someMethod() throws Exception{   }                  上边的代码又有什么问题?               很明显,上边的方法体是空的,它不实现任何的功能(没有一句代码),试问一个空方法体能抛出什么异常?当然Java并不阻止你这么干。最近,我也遇到类似的情景,方法声明会抛出异常,但是代码中并没有任何“机会”来“展示”异常。当我问开发员为什么要这样做的时候,他回答我说“我知道,它确实有点那个,但我以前就是这么干的并且它确实能为我工作。”               在C++社区曾经花了数年实践来实践如何使用异常,关于此类的争论在java社区才刚刚开始。我曾经看到许多Java程序员针对使用异常的问题进行争论。如果对于异常处理不当的话,异常可以大大减慢应用程序的执行速度,因为它将消耗内存和CPU来创建、抛出并捕获异常。如果过分的依赖异常处理,代码对易读和易使用这两方面产生影响,以至于会让我们写出上边两处“糟糕”代码。               异常原理                  大体上说,有三种不同的“情景”会导致异常的抛出:      l 编程错误导致异常(Exception due Programming errors): 这种情景下,异常往往处于编程错误(如:NullPointerException 或者 IllegalArgumentException),这时异常一旦抛出,客户端将变得无能为力。      l 客户端代码错误导致异常(Exception due client code errors): 说白点就是客户端试图调用API不允许的操作。      l 资源失败导致异常(Exception due to resource failures): 如内存不足或网络连接失败导致出现异常等。这些异常的出现客户端可以采取相应的措施来恢复应用程序的继续运行。               Java中异常的类型         Java 中定义了两类异常:      l Checked exception: 这类异常都是Exception的子类      l Unchecked exception: 这类异常都是RuntimeException的子类,虽然RuntimeException同样也是Exception的子类,但是它们是特殊的,它们不能通过client code来试图解决,所以称为Unchecked exception      举个例子,下图为NullPointerException的继承关系:            Figure 1. Sample exception hierarchy                  图中,NullPointerException继承自RuntimeException,所以它是Unchecked exception.               以往我都是应用checked exception多于Unchecked exception,最近,在java社区激起了一场关于checked exception和使用它们的价值的争论。这场争论起源于JAVA是第一个拥有Checked exception的主流OO语言这样一个事实,而C++和C#都是根本没有Checked exception,它们所有的异常都是unchecked。               一个checked exception强迫它的客户端可以抛出并捕获它,一旦客户端不能有效地处理这些被抛出的异常就会给程序的执行带来不期望的负担。               Checked exception还可能带来封装泄漏,看下面的代码:               public List getAllAccounts() throws    FileNotFoundException, SQLException{    ...   }                  上边的方法抛出两个异常。客户端必须显示的对这两种异常进行捕获和处理即使是在完全不知道这种异常到底是因为文件还是数据库操作引起的情况下。因此,此时的异常处理将导致一种方法和调用之间不合适的耦合。               接下来我会给出几种设计异常的最佳实践 (Best Practises for Designing the API)    1. 当要决定是采用checked exception还是Unchecked exception的时候,你要问自己一个问题,“如果这种异常一旦抛出,客户端会做怎样的补救?”      [原文:When deciding on checked exceptions vs. unchecked exceptions, ask yourself, "What action can the client code take when the exception occurs?"]      如果客户端可以通过其他的方法恢复异常,那么这种异常就是checked exception;如果客户端对出现的这种异常无能为力,那么这种异常就是Unchecked exception;从使用上讲,当异常出现的时候要做一些试图恢复它的动作而不要仅仅的打印它的信息,总来的来说,看下表:      Client's reaction when exception happens    Exception type      Client code cannot do anything    Make it an unchecked exception      Client code will take some useful recovery action based on information in exception    Make it a checked exception         此外,尽量使用unchecked exception来处理编程错误:因为unchecked exception不用使客户端代码显示的处理它们,它们自己会在出现的地方挂起程序并打印出异常信息。Java API中提供了丰富的unchecked excetpion,譬如:NullPointerException , IllegalArgumentException 和 IllegalStateException等,因此我一般使用这些标准的异常类而不愿亲自创建新的异常类,这样使我的代码易于理解并避免的过多的消耗内存。               2. 保护封装性(Preserve encapsulation)         不要让你要抛出的checked exception升级到较高的层次。例如,不要让SQLException延伸到业务层。业务层并不需要(不关心?)SQLException。你有两种方法来解决这种问题:      l 转变SQLException为另外一个checked exception,如果客户端并不需要恢复这种异常的话;      l 转变SQLException为一个unchecked exception,如果客户端对这种异常无能为力的话;      多数情况下,客户端代码都是对SQLException无能为力的,因此你要毫不犹豫的把它转变为一个unchecked exception,看看下边的代码:               public void dataAccessCode(){    try{    ..some code that throws SQLException    }catch(SQLException ex){    ex.printStacktrace();    }   }         上边的catch块紧紧打印异常信息而没有任何的直接操作,这是情有可原的,因为对于SQLException你还奢望客户端做些什么呢?(但是显然这种就象什么事情都没发生一样的做法是不可取的)那么有没有另外一种更加可行的方法呢?               public void dataAccessCode(){    try{    ..some code that throws SQLException    }catch(SQLException ex){    throw new RuntimeException(ex);    }   }                  上边的做法是把SQLException转换为RuntimeException,一旦SQLException被抛出,那么程序将抛出RuntimeException,此时程序被挂起并返回客户端异常信息。               如果你有足够的信心恢复它当SQLException被抛出的时候,那么你也可以把它转换为一个有意义的checked exception, 但是我发现在大多时候抛出RuntimeException已经足够用了。               3. 不要创建没有意义的异常(Try not to create new custom exceptions if they do not have useful information for client code.)         看看下面的代码有什么问题?      public class DuplicateUsernameException    extends Exception {}         它除了有一个“意义明确”的名字以外没有任何有用的信息了。不要忘记Exception跟其他的Java类一样,客户端可以调用其中的方法来得到更多的信息。      我们可以为其添加一些必要的方法,如下:      public class DuplicateUsernameException    extends Exception {    public DuplicateUsernameException    (String username){....}    public String requestedUsername(){...}    public String[] availableNames(){...}   }               在新的代码中有两个有用的方法:reqeuestedUsername(),客户但可以通过它得到请求的名称;availableNames(),客户端可以通过它得到一组有用的usernames。这样客户端在得到其返回的信息来明确自己的操作失败的原因。但是如果你不想添加更多的信息,那么你可以抛出一个标准的Exception:      throw new Exception("Username already taken");               更甚的情况,如果你认为客户端并不想用过多的操作而仅仅想看到异常信息,你可以抛出一个unchecked exception:      throw new RuntimeException("Username already taken");               另外,你可以提供一个方法来验证该username是否被占用。               很有必要再重申一下,checked exception应该让客户端从中得到丰富的信息。要想让你的代码更加易读,请倾向于用unchecked excetpion来处理程序中的错误(Prefer unchecked exceptions for all programmatic errors)。               4. Document exceptions.         你可以通过Javadoc’s @throws 标签来说明(document)你的API中要抛出checked exception或者unchecked exception。然而,我更倾向于使用来单元测试来说明(document)异常。不管你采用哪中方式,你要让客户端代码知道你的API中所要抛出的异常。这里有一个用单元测试来测试IndexOutOfBoundsException的例子:      public void testIndexOutOfBoundsException() {    ArrayList blankList = new ArrayList();    try {    blankList.get(10);    fail("Should raise an IndexOutOfBoundsException");    } catch (IndexOutOfBoundsException success) {}   }               上边的代码在请求blankList.get(10)的时候会抛出IndexOutOfBoundsException,如果没有被抛出,将fail("Should raise an IndexOutOfBoundsException")显示说明该测试失败。通过书写测试异常的单元测试,你不但可以看到异常是怎样的工作的,而且你可以让你的代码变得越来越健壮。                        下面作者将介绍界中使用异常的最佳实践(Best Practices for Using Exceptions)      1. 总是要做一些清理工作(Always clean up after yourself)      如果你使用一些资源例如数据库连接或者网络连接,请记住要做一些清理工作(如关闭数据库连接或者网络连接),如果你的API抛出Unchecked exception,那么你要用try-finally来做必要的清理工作:      public void dataAccessCode(){    Connection conn = null;    try{    conn = getConnection();    ..some code that throws SQLException    }catch(SQLException ex){    ex.printStacktrace();    } finally{    DBUtil.closeConnection(conn);    }   }      class DBUtil{    public static void closeConnection    (Connection conn){    try{    conn.close();    } catch(SQLException ex){    logger.error("Cannot close connection");    throw new RuntimeException(ex);    }    }   }               DBUtil是一个工具类来关闭Connection.有必要的说的使用的finally的重要性是不管程序是否碰到异常,它都会被执行。在上边的例子中,finally中关闭连接,如果在关闭连接的时候出现错误就抛出RuntimeException.               2. 不要使用异常来控制流程(Never use exceptions for flow control)         下边代码中,MaximumCountReachedException被用于控制流程:      public void useExceptionsForFlowControl() {    try {    while (true) {    increaseCount();    }    } catch (MaximumCountReachedException ex) {    }    //Continue execution   }      public void increaseCount()    throws MaximumCountReachedException {    if (count >= 5000)    throw new MaximumCountReachedException();   }               上边的useExceptionsForFlowControl()用一个无限循环来增加count直到抛出异常,这种做法并没有说让代码不易读,但是它是程序执行效率降低。         记住,只在要会抛出异常的地方进行异常处理。                  3. 不要忽略异常         当有异常被抛出的时候,如果你不想恢复它,那么你要毫不犹豫的将其转换为unchecked exception,而不是用一个空的catch块或者什么也不做来忽略它,以至于从表面来看象是什么也没有发生一样。                  4. 不要捕获顶层的Exception         unchecked exception都是RuntimeException的子类,RuntimeException又继承Exception,因此,如果单纯的捕获Exception,那么你同样也捕获了RuntimeException,如下代码:         try{   ..   }catch(Exception ex){   }         一旦你写出了上边的代码(注意catch块是空的),它将忽略所有的异常,包括unchecked exception.





J2EE项目中的异常处理

1. checked 和 unchecked exceptions
   致命错误, 不可恢复错误, 第三方包错误 包装为unchecked exception。
   业务逻辑异常流,安全异常包装为checked exception. J2ee应用中有Business exception,

DB exception, security exception, confirmation exception

2. 抛出异常的策略
   2.1 采用J2EE应用程序的 前端控制器(front controller) 集中控制处理,开发者实现具体的

业务逻辑的时候可以不关心异常的捕捉
   2.2 方法签名中throws 一个基类的异常。这样前端控制器中只要捕捉这个基类的异常,开发者

可以定义自己派生的异常具体类

3. 在web层处理unchecked exceptions
   所有的unchecked exceptions 都应该在web层进行处理, 可以在web.xml中配置异常发生时显

示定制的错误页面。

4. 封装第三方包的所有异常(Exception)为一个我们定义的unchecked exception

5. 对来自数据库操作的checked exception, 比如Employee not found, Order not found可以抽

象一个RecordNotFoundException. 至于这个RecordNotFoundException是因为Employee还是Order

引起的作为context, 由context一起组成error code,抛给前端控制器。

6. 前端控制器中的错误处理
   基类的异常处理(应用模板方法设计模式 Template Method)和子类(实现业务)的部分代码如

下,以struts为例

public abstract class BaseAppDispatchAction
extends DispatchAction{
...
protected static ThreadLocal
expDisplayDetails = new ThreadLocal();

public ActionForward execute(
ActionMapping mapping,
ActionForm form,
HttpServletRequest request,
HttpServletResponse response) throws Exception{
    ...
    try{
        String actionMethod = request
           .getParameter(mapping.getParameter());
        finalDestination =dispatchMethod(mapping,
          form, request, response,actionMethod);
    }catch (BaseAppException Ex) {
        ExceptionDisplayDTO expDTO =
           (ExceptionDisplayDTO)expDisplayDetails
               .get();
        IExceptionHandler expHandler =
            ExceptionHandlerFactory
                .getInstance().create();
        ExceptionDTO exDto = expHandler
            .handleException(
                expDTO.getContext(), userId, Ex);
        ActionErrors errors = new ActionErrors();
        errors.add(ActionErrors.GLOBAL_ERROR,
            new ActionError(exDto
                .getMessageCode()));
        saveErrors(request, errors);
        return mapping.findForward(expDTO
            .getActionForwardName());
    } catch(Throwable ex){
           //log the throwable
           //throw ex;
    } finally {
           expDisplayDetails.set(null);
    }

    业务子类中代码如下:
...
String exceptionActionForward =
    "SearchAdjustmentPage";
String exceptionContext =
    "divisionAction.searchDivision";
       
ExceptionDisplayDTO expDTO =
    new ExceptionDisplayDTO(expActionForward,
        exceptionContext);
expDisplayDetails.set(expDTO);
...
DivisionDTO divisionDTO =divisionBusinessDelegate
   .getDivisionByNum(fromDivisionNum);
...

ExceptionDisplayDTO 作为data transfer object放在ThreadLocal中专门为了父类的异常处理



7. Exception Handler
    上面的集中异常控制是在Excepiton Handler中处理的,包括:
    根据异常类型获取异常代码,异常代码会用来显示异常信息. 异常类型与异常代码的对应关系

可以放到一个XML文件中. 根据异常代码取异常信息可以由ResouceBundle来实现,这样同时解决了

异常信息国际化的问题. 两个异常配置的xml文件如下:
<exception name="EmployeeConfirmationException">
    <messagecode>messagecode.employeeconfirmation</messagecode>
    <confirmationind>true</confirmationind>
    <loggingtype>nologging</loggingtype>
</exception>
<exception name="RecordNotFoundException">
    <messagecode>messagecode.recordnotfound</messagecode>
    <confirmationind>false</confirmationind>
    <contextind>true</contextind>
    <loggingtype>error</loggingtype>
</exception>
第二个异常文件考虑了我们上面讲的context问题.     如果抛出的异常是

RecordNotFoundException, 并且当前的context是order.getOrder , 那么error code就

是:messagecode.recordnotfound.order.getOrder

为了避免多次解析这个XML文件,可以实现一个ExceptionInfoCache ,启动应用时缓冲这个xml的内

容.

ExceptionHandler中handleException方法代码片断如下:

public ExceptionDTO handleException(String userId,
      BaseAppException exp) {
    ExceptionDTO exDTO = new ExceptionDTO();
    ExceptionInfoCache ecache =
     ExceptionInfoCache.getInstance();
    ExceptionInfo exInfo = ecache
      .getExceptionInfo(
        ExceptionHelper.getClassName(exp));
    String loggingType = null;
    if (exInfo != null) {
        loggingType = exInfo.getLoggingType();
        exDTO.setConfirmation(exInfo
            .isConfirmation());
        exDTO.setMessageCode(exInfo
            .getMessageCode());
    }

    FileLogger logger = new FileLoggerFactory()
        .create();
    logger.logException(exp, loggingType);




文章出处:飞诺网(www.firnow.com):http://dev.firnow.com/course/3_program/java/javajs/20090407/164574_2.html

 

分享到:
评论

相关推荐

    java异常处理习题

    Java 异常处理习题 Java 异常处理是 Java 编程语言中的一种重要机制,用于处理程序在运行时可能出现的错误或异常情况。下面是关于 Java 异常处理的习题和知识点总结: 一、Java 异常处理关键字 * Java 中用来抛出...

    java异常处理java异常处理

    java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理java异常处理

    Java异常处理

    ### Java异常处理 #### 一、异常的基本概念 在编程领域中,异常是指程序在运行过程中遇到的问题或错误状态,这些问题通常会导致程序无法按照预期的方式继续执行。在Java语言中,异常处理是一种重要的机制,用于...

    java 异常处理ppt

    Java异常处理是编程中至关重要的一个环节,它允许开发者优雅地处理程序运行时可能出现的错误,确保程序的健壮性和稳定性。Java中的异常处理机制基于一个独特的概念——异常(Exception),这是一种表示程序运行中...

    Java异常处理体系

    这个主题,"Java异常处理体系",涉及到Java如何优雅地处理程序运行时的异常,确保程序的健壮性和稳定性。这篇博文链接(已提供但无法直接访问)可能详细解释了Java异常处理的基本概念、机制以及最佳实践。 Java异常...

    浅析JAVA异常处理机制.pdf

    ### 浅析JAVA异常处理机制 #### 一、Java异常处理机制概述 异常处理是Java语言中的一个重要机制,它能够确保程序在遇到不可预料的情况时仍能维持稳定运行。异常处理主要包括三个方面:捕获异常、控制程序流程以及...

    java 异常处理 代码实例

    Java异常处理是编程中至关重要的一个环节,它确保了程序在遇到错误时能够优雅地运行,而不是突然崩溃。本篇文章将深入探讨Java异常处理的概念、机制以及如何通过代码实例进行有效的异常管理。 Java异常处理的核心...

    Java异常处理终结篇——如何进行Java异常处理设计 - 望远的个人页面 - 开源中国社区1

    Java异常处理设计是Java编程中一个至关重要的环节,它直接影响到程序的稳定性和可维护性。在Java中,异常处理是通过try-catch-finally语句块来实现的,主要涉及两大类异常:编译时异常(Checked Exception)和运行时...

    Java 异常处理的 9 个最佳实践

    Java异常处理的9个最佳实践涵盖了开发者在进行Java编程时应当遵循的一系列规则和方法,以确保代码的健壮性和可维护性。异常处理是编程中的一个重要部分,它能够帮助我们更有效地管理和响应程序运行时的错误情况。 ...

    Java异常处理,非常适合Java爱好者

    Java异常处理是Java编程语言中的一个关键特性,它允许程序员优雅地处理程序运行时可能出现的错误和不正常状况。在Java中,异常是程序执行过程中出现的问题,它们中断了正常的控制流程,使得程序无法继续按照预期的...

    java高端技术课程--Java异常处理详解

    Java异常处理是Java编程中一个至关重要的概念,它关乎到程序的健壮性和稳定性。在Java语言中,异常处理提供了一种机制,使得程序在遇到错误或非正常情况时能够优雅地处理问题,而不是突然崩溃。这个“java高端技术...

    高效java异常处理机制

    通过遵循上述最佳实践,开发者可以编写出更健壮、可维护的Java代码,有效地利用Java异常处理机制来提高程序的稳定性。同时,阅读并理解"Effective Java Exceptions"文档可以帮助进一步深化对Java异常处理的理解。

    java异常处理例题代码.pdf

    从给定的文件内容中可以看出,这是一系列Java程序示例,它们用来说明Java异常处理机制。异常处理是Java语言中一个非常重要的部分,它能够帮助程序在遇到错误情况时仍然可以正常运行或者优雅地退出。这些程序通过各种...

    java异常处理练习题

    java异常练习题,北大青鸟学士后课后练习,主要是异常处理的小测验

    Java 异常处理的误区和经验总结

    以下是对Java异常处理的一些误区和经验总结。 **误区一:过度使用try-catch块** 有些开发者习惯于在每个函数的开始部分都套用try-catch块,以为这样可以捕捉所有可能出现的异常。实际上,这种做法使得代码变得混乱...

    零基础学Java异常处理

    零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理零基础学Java异常处理...

    Java异常处理总结

    ### Java异常处理总结 #### 一、Java异常的基础知识 在Java编程中,异常处理是一项极其重要的技术。异常处理能够帮助开发者优雅地处理程序运行时遇到的问题,避免程序崩溃,并提供给用户友好的反馈信息。 **什么...

Global site tag (gtag.js) - Google Analytics