展望2011

2007-01-15 | java的异常处理(1)【转】

上一篇 / 下一篇  2007-04-27 22:00:56 / 个人分类:编程基础

0C-Tn!bf?0浅析Java语言中两种异常的差别51Testing软件测试网.MC!{2{1So+J/q#v:^

51Testing软件测试网Z%J+U)sns*M'L

Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。51Testing软件测试网2uZ/{)O i8[ u2] `

1u0E5^"n;_7hig0  它们的不同之处表现在两方面:机制上和逻辑上。51Testing软件测试网5]}$SO/sumQ

51Testing软件测试网2L)b K Q0[^/pN

  一、机制上

}&|C3f3p#L:U:I/b&I051Testing软件测试网3p&zR)Km-g

  它们在机制上的不同表现在两点:1.如何定义方法;2. 如何处理抛出的异常。请看下面CheckedException的定义:

6u"u/Z p!\s[051Testing软件测试网}f QZ%Lj

  public class CheckedException extends Exception
j`1_ ~\/A|([^2h0  {
fbD\BD0  public CheckedException() {}51Testing软件测试网 q S0B%F ^
  public CheckedException( String message )51Testing软件测试网6m ]5I$E@(EROx
  {51Testing软件测试网t.Y?1W {u}
  super( message );
Kw Ywk0  }51Testing软件测试网{D]A.y P yE
  }

J1Rx*|]gd)Q051Testing软件测试网/w8m(s G6H m M

  以及一个使用exception的例子:

V*r hl%s A9K%V7_0

~f,K8o cv]0  public class ExceptionalClass51Testing软件测试网-I"|w6V k7D'g2|I
  {
:v#j-N^%]DRv0  public void method1()
H z{4]-wW#f!@0  throws CheckedException51Testing软件测试网0MCr,blKE
  {51Testing软件测试网*E#d$HqP;\ TI;Z
   // ... throw new CheckedException("...出错了");
XXGQZ7|({0  }51Testing软件测试网#T e!Z.cl YUs
  public void method2( String arg )51Testing软件测试网,h^ rS[Y
  {51Testing软件测试网&t P2k{d*q
   if( arg == null )
(o$cHoY g.G#o\6vmh!M0   {51Testing软件测试网6A9PT[2Ku6s/IYz
    throw new NullPointerException("method2的参数arg是null!");
O ~P5X2|i3S0   }51Testing软件测试网yxX3H"{4q%Q_5Z
  }
"xh-k1]d2v0  public void method3() throws CheckedException51Testing软件测试网y(@8VTOT/A7{s"~
  {51Testing软件测试网$m \$ws"yX
   method1();51Testing软件测试网H G.k:BH
  }
as/GI ?0  }
51Testing软件测试网/M/j)f.Ah;~

51Testing软件测试网3hH1b ?0E ~rgV

  你可能已经注意到了,两个方法method1()和method2()都会抛出exception,可是只有method1()做了声明。另外,method3()本身并不会抛出exception,可是它却声明会抛出CheckedException。在向你解释之前,让我们先来看看这个类的main()方法:51Testing软件测试网m&oBf'q[M

h:N;d5?)oH0T4C0  public static void main( String[] args )51Testing软件测试网%I T+e,Rn
  {
dH+OX:r7lEW3G0  ExceptionalClass example = new ExceptionalClass();51Testing软件测试网2k0`:X"R7{#D \ js-Z
  try51Testing软件测试网-A5{;\k5j2~
  {
D%?X;@Jtr6^Riy0  example.method1();51Testing软件测试网JVK0Tv9C7x v
  example.method3();
7Sj&~)c%`o um0  }
5[ }-zB:O/vyVqG0  catch( CheckedException ex ) { } example.method2( null );
'Q8Lvtk3_+n0I0  }

oB!B.b5V `5Pe!n051Testing软件测试网 X O-B#w Qyf0i

  在main()方法中,如果要调用method1(),你必须把这个调用放在try/catch程序块当中,因为它会抛出Checked exception。51Testing软件测试网'ms6~'Oaqtv

51Testing软件测试网)MH j+] pi@tkW}]"G

相比之下,当你调用method2()时,则不需要把它放在try/catch程序块当中,因为它会抛出的exception不是checked exception,而是runtime exception。会抛出runtime exception的方法在定义时不必声明它会抛出exception。

*Br,h[rJ0EIQ*CZH Y0

4_| d#mO9G Y@0  现在,让我们再来看看method3()。它调用了method1()却没有把这个调用放在try/catch程序块当中。它是通过声明它会抛出method1()会抛出的exception来避免这样做的。它没有捕获这个exception,而是把它传递下去。实际上main()方法也可以这样做,通过声明它会抛出Checked exception来避免使用try/catch程序块(当然我们反对这种做法)。

#{Q9u'eY6?h7|bl051Testing软件测试网tfN[u}E+}

  小结一下:51Testing软件测试网 T&B @c } PQ?t0WD

tp~/H0}0  * Runtime exceptions:51Testing软件测试网]V?'h.|,a

51Testing软件测试网2fgVTu~M&iA

  在定义方法时不需要声明会抛出runtime exception;51Testing软件测试网 l W jL3v

51Testing软件测试网9I8F/mI9y:g p+J

  在调用这个方法时不需要捕获这个runtime exception;

az#ht!o_K051Testing软件测试网&yC5E#Z&Oi

  runtime exception是从java.lang.RuntimeException或java.lang.Error类衍生出来的。51Testing软件测试网'R,SE.Bj3?@o]

\1`+F-w p5z-_~ }0  * Checked exceptions:51Testing软件测试网0b1^7Nl&|1C.r(MJ6l_:]

51Testing软件测试网9o ?O ~B

  定义方法时必须声明所有可能会抛出的checked exception;

U\{nsyZn051Testing软件测试网3B_|tP`p

  在调用这个方法时,必须捕获它的checked exception,不然就得把它的exception传递下去;51Testing软件测试网~[?|a5s1GH I)@/?

51Testing软件测试网 T'c`6v:O`I

  checked exception是从java.lang.Exception类衍生出来的。

W^`L6aeM051Testing软件测试网^1t a9U*K*{z]d s!_

  二、逻辑上

;xT%n%u*|N051Testing软件测试网s-i}q!W]:]

  从逻辑的角度来说,checked exceptions和runtime exception是有不同的使用目的的。checked exception用来指示一种调用方能够直接处理的异常情况。而runtime exception则用来指示一种调用方本身无法处理或恢复的程序错误。

2U6Zv;N$F051Testing软件测试网J7\1HIx*A'z

  checked exception迫使你捕获它并处理这种异常情况。以java.net.URL类的构建器(constructor)为例,它的每一个构建器都会抛出MalformedURLException。MalformedURLException就是一种checked exception。设想一下,你有一个简单的程序,用来提示用户输入一个URL,然后通过这个URL去下载一个网页。如果用户输入的URL有错误,构建器就会抛出一个exception。既然这个exception是checked exception,你的程序就可以捕获它并正确处理:比如说提示用户重新输入。

-SD3A%CiE0

+fp a^0ZHO:S0  再看下面这个例子:

'L?|7rF+N G051Testing软件测试网J{[!fJ L/}G

  public void method()51Testing软件测试网.ll3RZnk$@w
  {
%gVk/?.d#x;eq`I0  int [] numbers = { 1, 2, 3 };
:rn-`R9K4{;k0  int sum = numbers[0] numbers[3];51Testing软件测试网3S%~%n(b|
  }

#f%U;I p(T7? XQ j051Testing软件测试网{{5e&t BH

  在运行方法method()时会遇到ArrayIndexOutOfBoundsException(因为数组numbers的成员是从0到2)。对于这个异常,调用方无法处理/纠正。这个方法method()和上面的method2()一样,都是runtime exception的情形。上面我已经提到,runtime exception用来指示一种调用方本身无法处理/恢复的程序错误。而程序错误通常是无法在运行过程中处理的,必须改正程序代码。

1j3g"LV1zi051Testing软件测试网 V"q#tDqiD#?}

  总而言之,在程序的运行过程中一个checked exception被抛出的时候,只有能够适当处理这个异常的调用方才应该用try/catch来捕获它。而对于runtime exception,则不应当在程序中捕获它。如果你要捕获它的话,你就会冒这样一个风险:程序代码的错误(bug)被掩盖在运行当中无法被察觉。因为在程序测试过程中,系统打印出来的调用堆栈路径(StackTrace)往往使你更快找到并修改代码中的错误。有些程序员建议捕获runtime exception并纪录在log中,我反对这样做。这样做的坏处是你必须通过浏览log来找出问题,而用来测试程序的测试系统(比如Unit Test)却无法直接捕获问题并报告出来。51Testing软件测试网 ?/k4?|-jD

51Testing软件测试网'cF0@ S u5J

  在程序中捕获runtime exception还会带来更多的问题:要捕获哪些runtime exception?什么时候捕获?runtime exception是不需要声明的,你怎样知道有没有runtime exception要捕获?你想看到在程序中每一次调用方法时,都使用try/catch程序块吗?

i!c{K,h%L0

TAG: JAVA 异常处理 编程基础

 

评分:0

我来说两句

Open Toolbar