Board logo

标题: J2SE综合:浅析Java语言中两种异常的差别 [打印本页]

作者: qingqing3721    时间: 2011-8-8 23:53     标题: J2SE综合:浅析Java语言中两种异常的差别

Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。
  它们的不同之处表如今两方面:机制上和逻辑上。
  一、机制上
  它们在机制上的不同表如今两点:1.如何定义方法;2. 如何处理抛出的异常。请看下面CheckedException的定义:
  public class CheckedException extends Exception  {
  public CheckedException() {}  public CheckedException( String message )  {  super( message );  }  }
  以及一个使用exception的例子:
  public class ExceptionalClass  {
  public void method1()  throws CheckedException  {   // ... throw new CheckedException( “...出错了“ );  }  public void method2( String arg )  {   if( arg == null )   {    throw new NullPointerException( “method2的参数arg是null!” );   }  }  public void method3() throws CheckedException  {   method1();  }  }
  你能够曾经注意到了,两个方法method1()和method2()都会抛出exception,可是只要method1()做了声明。另外,method3()自身并不会抛出exception,可是它却声明会抛出CheckedException。在向你解释之前,让我们先来看看这个类的main()方法:
  public static void main( String[] args )  {
  ExceptionalClass example = new ExceptionalClass();  try  {  example.method1();  example.method3();  }  catch( CheckedException ex ) { } example.method2( null );  }
  在main()方法中,假如要调用method1(),你必须把这个调用放在try/catch程序块当中,由于它会抛出Checked exception。
  相比之下,当你调用method2()时,则不需要把它放在try/catch程序块当中,由于它会抛出的exception不是checked exception,而是runtime exception。会抛出runtime exception的方法在定义时不用声明它会抛出exception。
  如今,让我们再来看看method3()。它调用了method1()却没有把这个调用放在try/catch程序块当中。它是经过声明它会抛出 method1()会抛出的exception来防止这样做的。它没有捕捉这个exception,而是把它传递下去。实践上main()方法也可以这样做,经过声明它会抛出Checked exception来防止使用try/catch程序块(当然我们支持这种做法)。
  小结一下:
  * Runtime exceptions:
  在定义方法时不需要声明会抛出runtime exception;
  在调用这个方法时不需要捕捉这个runtime exception;
  runtime exception是从java.lang.RuntimeException或java.lang.Error类衍生出来的。
  * Checked exceptions:
  定义方法时必须声明所有能够会抛出的checked exception;
  在调用这个方法时,必须捕捉它的checked exception,不然就得把它的exception传递下去;
  checked exception是从java.lang.Exception类衍生出来的。
  二、逻辑上
  从逻辑的角度来说,checked exceptions和runtime exception是有不同的使用目的的。checked exception用来指示一种调用方可以直接处理的异常情况。而runtime exception则用来指示一种调用方自身无法处理或恢复的程序错误。
  checked exception迫使你捕捉它并处理这种异常情况。以java.URL类的构建器(constructor)为例,它的每一个构建器都会抛出 MalformedURLException。MalformedURLException就是一种checked exception。想象一下,你有一个简单的程序,用来提示用户输入一个URL,然后经过这个URL去下载一个网页。假如用户输入的URL有错误,构建器就会抛出一个exception。既然这个exception是checked exception,你的程序就可以捕捉它并正确处理:比方说提示用户重新输入。
  再看下面这个例子:
  public void method()  {
int [] numbers = { 1, 2, 3 };  int sum = numbers[0] numbers[3];  }
  在运行方法method()时会遇到ArrayIndexOutOfBoundsException(由于数组numbers的成员是从0到2)。关于这个异常,调用方无法处理/纠正。这个方法method()和下面的method2()一样,都是runtime exception的情形。下面我曾经提到,runtime exception用来指示一种调用方自身无法处理/恢复的程序错误。而程序错误通常是无法在运行过程中处理的,必须矫正程序代码。
  总而言之,内衣在程序的运行过程中一个checked exception被抛出的时候,只要可以适当处理这个异常的调用刚才应该用try/catch来捕捉它。而关于runtime exception,则不应当在程序中捕捉它。假如你要捕捉它的话,你就会冒这样一个风险:程序代码的错误(bug)被掩盖在运行当中无法被发觉。由于在程序测试过程中,系统打印出来的调用堆栈路径(StackTrace)往往使你更快找到并修正代码中的错误。有些程序员建议捕捉runtime exception并纪录在log中,我支持这样做。这样做的坏处是你必须经过阅读log来找出成果,而用来测试程序的测试系统(比方Unit Test)却无法直接捕捉成果并报告出来。
  在程序中捕捉runtime exception还会带来更多的成果:要捕捉哪些runtime exception?什么时候捕捉?runtime exception是不需要声明的,你怎样知道有没有runtime exception要捕捉?你想看到在程序中每一次调用方法时,都使用try/catch程序块吗?




欢迎光临 编程开发论坛 (http://bbs.lihuasoft.net/) Powered by Discuz! 6.0.0