`
noknower
  • 浏览: 120167 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

避免在Java中使用Checked Exception

    博客分类:
  • JAVA
阅读更多
这篇文章指出了Java中checked Exception的一些缺点,提出应该在程序设计中避免使用checked Exception,对于需要处理checked Exception的代码,可以使用ExceptionAdapter这个类对checked Exception进行包装。这篇文章的概念和ExceptionAdapter这个类均源自Bruce Eckel的Does Java need Checked Exception。


Java的Exception分为两类,一类是RuntimeException及其子类,另外一类就是checked Exception。Java要求函数对没有被catch处理掉的checked Exception,需要将其写在函数的声明部分。然而,这一要求常常给程序员带来一些不必要的负担。



为了避免在函数声明中写throws部分,在Java项目里面常常可以看到以下代码用来‘吞掉’Exception:
       try {

           // ...

       } catch (Exception ex) {

           ex.printStackTrace();

       }




这显然不是一个好的处理Exception办法,事实上,catch并处理一个Exception意味着让程序从发生的错误(Exception)中恢复过来。从这种意义上说,已上的代码只可能在一些很简单的情况下工作而不带来问题。



对于很多Exception,往往没有去处理它并让程序从错误中恢复出来的办法,这时唯一能做的事情可能就是在界面上显示一些提示信息给用户。这种情况下让程序抛出遇到的Exception是更为合理的做法。然而,这样做会使得一些函数的声明急剧膨胀。一个函数可能需要声明会抛出的7、8个checked Exception,而且每个调用它的函数也需要同样的声明。



比这更糟糕的是,这有可能破坏类设计的open-close原则。简单来说,open-close原则是指当扩展一个模块的时候,可以不影响其现有的client。open-close原则是通过继承来实现的,当继承一个类的时候,我们既扩展了这个类,也不会影响原有的client(因为对这个类没有改动)。



现在考虑下面这种情况,有一个父类Base:

public class Base {

   

    public void foo() throws ExceptionA {

       // ...

    }

}




现在需要继承Base这个类并重载foo这个方法,在新的实现中,foo可能抛出ExceptionB:
public class Extend extends Base {

   

    public void foo() throws ExceptionB {

       // ...

    }

}




然而,这样写在Java里面是不合法的,因为Java把可能会抛出的Exception看作函数特征的一部分,子类声明抛出的Exception必须是父类的子集。



可以在Base类的foo方法中加入抛出ExceptionB的声明,然而,这样就破坏了open-close原则。而且,有时我们没有办法去修改父类,比如当重载一个Jdk里的类的时候。



另一个可能的做法是在Extend的foo方法中catch住ExceptionB,然后构造一个ExceptionA并抛出。这是个可行的办法但也只是一个权宜之计。



如果使用RuntimeException,这些问题都不会存在。这说明checked Exception并不是一个很实用的概念,也意味着在程序设计的时候,我们应该让自己的Exception类继承RuntimeException而不是Exception。(这和JDK的建议正好相反,但实践证明这样做代码的质量更好。)



对于那些需要处理checked Exception的代码,可以利用一个ExceptionAdapter的类把checked Exception包装成一个RuntimeException抛出。ExceptionAdapter来自Bruce Eckel的Does Java need Checked Exception这篇文章,在这里的ExceptionAdapter是我根据JDK 1.4修改过的:

public class ExceptionAdapter extends RuntimeException {

    

    public ExceptionAdapter(Exception ex) {

       super(ex);

    }

    

    public void printStackTrace(java.io.PrintStream s) { 

       getCause().printStackTrace(s); 

    }

    

    public void printStackTrace(java.io.PrintWriter s) { 

       getCause().printStackTrace(s);

    }

    

    // rethrow()的作用是把被包装的Exception再次抛出。

    public void rethrow() 

       throws Exception

    {

       throw (Exception) getCause();

    }

}




分享到:
评论

相关推荐

    JSTL详细标签库介绍

    target=_blank>避免在Java中使用Checked Exception</A> <LI><A title="开源 JVM 一览" href="http://www.jspcn.net/htmlnews/11049386275931969.html" target=_blank>开源 JVM 一览</A> <LI id=more1><A ...

    Java中Error与Exception的区别.doc

    ### Java中Error与Exception的区别 #### 一、概述 ...在开发过程中,应当尽量避免抛出 `Error` 类型的异常,并通过合理的异常处理策略来管理 `Exception` 类型的异常,从而构建更加稳定和可靠的软件系统。

    java 异常 问题收集 Exception

    在方法声明中使用throws关键字可以声明该方法可能抛出的检查性异常,这样调用者就必须处理这些异常,或者继续在上层抛出。 5. try-with-resources语句: 自Java 7开始,引入了try-with-resources语句,它能自动...

    JAVA Exception Handling & UI Design

    本项目中,"JAVA Exception Handling & UI Design" 涉及了如何在用户界面(UI)设计中结合异常处理,以及使用HashMap这一数据结构。 首先,我们来看看异常处理。Java中的异常分为两种类型:Checked异常和Unchecked...

    java exception

    Java异常是程序运行时出现的错误情况,它中断了正常的代码执行流程。...在实际开发中,应该根据具体情况选择合适的异常处理策略,避免过度使用`try-catch`,同时确保对异常的处理能够提供足够的信息以便调试。

    Java中Error和Exception的区别.pdf

    在实际编程中,应该尽量避免让程序抛出`Error`,因为这通常意味着系统层面的问题,而`Exception`则应当被适当地捕获和处理,以提高程序的稳定性和用户体验。通过良好的异常处理,开发者可以确保程序在面对异常情况时...

    Java精华(免费版)

    编程实例:在java程序中启动一个windows记事本程序的运行实例,并在该运行实例中打开该运行程序的源文件,启动的记事本程序5秒后关闭。 public class Property { public static void main(String[] args) { Process ...

    Java常见基础知识总结

    Checked Exception即受检查异常,Java代码在编译过程中,如果受检查异常没有被catch或者throws关键字处理的话,就没办法通过编译。Unchecked Exception即不受检查异常,Java代码在编译过程中,我们即使不处理不受...

    error与Exception的区别

    在编程世界中,错误处理是不可或缺的一部分,Java和许多其他编程语言中,"error"和"exception"这两个术语经常被提及。理解它们之间的区别对于编写健壮和可靠的代码至关重要。 首先,我们来看"error"。在Java中,`...

    Java Exception 几种不适当的处理

    尽管在C语言中使用返回值来指示函数的执行状态是一种常见做法,但在Java中,这通常被视为不良实践。利用函数返回值(如布尔值或整数值)来表示操作的成功或失败,存在以下问题:返回信息过于简单,无法携带丰富的...

    Java编程异常处理最佳实践【推荐】

    在try中使用了一个资源,比如InputStream,之后需要关闭它。在这种情况下,一个常见的错误是在try的末尾关闭了资源。但是,这种方法存在问题,只要不抛出异常,这种方法就可以很好地运行。但是你在try里调用了一个或...

    通过实例了解java checked和unchecked异常

    checked 异常继承自 java.lang.Exception,需要在代码中显式地通过 try-catch 捕获或者再抛出。如果不需要对这个异常做处理,可以简地将异常再次抛出。但是,这种做法存在一些不足之处,例如代码变得冗长、空的 ...

    exception 异常处理 exception

    ### 异常处理知识点解析 #### 一、异常处理概念 ...通过上述分析,我们可以看到,在Java中合理地使用异常处理机制是非常重要的,它不仅能够帮助我们编写更加健壮的程序,还能提高代码的可读性和可维护性。

    浅谈Java异常.docx

    根据《Effective Java》的建议,对于可以恢复的条件,应该使用`CheckedException`,而对于程序错误或不可恢复的情况,使用`RuntimeException`更为合适。 在处理异常时,如果抛出的是`CheckedException`,调用该方法...

    imooc_exception_book_java_exception_Book2_租车_源码.zip

    通过分析这份源码,我们可以深入理解Java异常处理机制以及如何在实际项目中应用这些概念。下面将详细探讨相关知识点。 1. **Java异常体系结构**: Java中的异常都是`java.lang.Throwable`类的子类,分为Error和...

    Java程序设计 7 异常处理.pptx

    如果在方法中可能出现异常,就需要在方法声明中使用 throws 语句来声明可能出现的异常类型。 异常的抛出 在 Java 中,异常可以由应用程序本身产生,也可以由 Java 虚拟机产生。异常的抛出可以通过 throw 语句来...

    关于java的一些小异常的处理

    `Checked Exception`类似于需要你直接面对并处理的情况,即如果你知道如何解决或者可以预测到某种错误的发生,你应该使用`Checked Exception`,并且在调用方法的代码中处理这些异常。这确保了调用者不能忽视可能发生...

    java异常(Exception)处理机制详解

    异常的使用可以分为两类:CheckedException 和 UncheckedException。 CheckedException 需要用 try...catch... 显示的捕获,而 UncheckedException 不需要捕获。 例如,IOException 是一个 Checked Exception,需要...

    java中高级面试题十大总结

    - Java中的异常分为检查异常(Checked Exception)和运行时异常(Unchecked Exception),前者在编译阶段必须处理,后者在运行时抛出。 - try-catch-finally语句块用于捕获和处理异常,finally块确保关键代码总会...

    JAVA实验九异常处理.pdf

    在程序中使用用户输入之前应该进行适当的验证,并准备好相应的异常处理逻辑,以避免程序因不当输入而出错。 通过上面的知识点,可以看出异常处理在Java程序设计中的重要性,良好的异常处理可以提高程序的健壮性和...

Global site tag (gtag.js) - Google Analytics