展望2011

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

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

51Testing软件测试网7B8V}@'n$i

浅析Java语言中两种异常的差别51Testing软件测试网0\O+t1X?2Y,`w

51Testing软件测试网];C6Q;dL+n

Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。

-I8c0y_)~0

(o[DW,_9T)}2[0  它们的不同之处表现在两方面:机制上和逻辑上。51Testing软件测试网 A&]3t$q&[sjl

9NH\[ B F,oA0  一、机制上

9J&j,OD,N1m0

*m`0s!x(q0  它们在机制上的不同表现在两点:1.如何定义方法;2. 如何处理抛出的异常。请看下面CheckedException的定义:51Testing软件测试网G5r S2Aj"Q(?*}g.gI

51Testing软件测试网2uTu j8L'N5[%|#}

  public class CheckedException extends Exception51Testing软件测试网hp8G~U D#nei)H
  {51Testing软件测试网)l(v'pv%Y/X4?i
  public CheckedException() {}
2?%cQ ~B}0  public CheckedException( String message )51Testing软件测试网)@#`Iu4Ky6n)M&A
  {51Testing软件测试网 AY9xb{d_W:v0?Vt
  super( message );
_8CY%p3bb.f-r3B0D2I0  }
wnu*J[[O`D*H0  }
51Testing软件测试网 Y(IQ(t Oi4W

n6h#f4Q,l9B j0  以及一个使用exception的例子:

DGm-x a[s%g051Testing软件测试网o ]P;[y2z,z|

  public class ExceptionalClass
#D ]7C RT9_0  {
qB1c/?;X!`3Q2z0  public void method1()
Dk!I5mHrp1@%Xn0  throws CheckedException
`dWs!q0  {51Testing软件测试网K VJT%kd
   // ... throw new CheckedException("...出错了");51Testing软件测试网^$nkSQ6G0U!^
  }
Id i'Tw,^ DS0  public void method2( String arg )51Testing软件测试网J@] ev~
  {
$L'k2hU8L!~ _D0   if( arg == null )
^h-ws/??0   {51Testing软件测试网!Aq(xC*U;\}*wd e
    throw new NullPointerException("method2的参数arg是null!");
"V Qf+QI4oBq-B0   }
5A L%|:B[(~8[0  }51Testing软件测试网G+S[?Z7tH
  public void method3() throws CheckedException51Testing软件测试网8V,vWybn
  {51Testing软件测试网cG!V2g/S)g
   method1();51Testing软件测试网8P#}$l6Q&r
  }
c&K1S7P Um9_0  }

3Xq%W&\*K&|0

Qf S%|H-y9]m0  你可能已经注意到了,两个方法method1()和method2()都会抛出exception,可是只有method1()做了声明。另外,method3()本身并不会抛出exception,可是它却声明会抛出CheckedException。在向你解释之前,让我们先来看看这个类的main()方法:

~H&h,@GtB0

zp$v9S,FqKW`#u#W0  public static void main( String[] args )51Testing软件测试网N7l,Jm"LJj R1UE3]9k
  {
'P3B[q Uu;E+xD0  ExceptionalClass example = new ExceptionalClass();51Testing软件测试网` N2O(V7ve?3dc:xJo
  try
5?!YbLS8s)n#J0  {
*O8ntoN0  example.method1();
K:w cV_*@0  example.method3();
"uf+b,i/pv0  }
Lmln;HfS&w#L/wD0  catch( CheckedException ex ) { } example.method2( null );
\oSfz*bv;?m0  }

'` [ I8j7g;S!u051Testing软件测试网:Z-N%A;w i6A4m

  在main()方法中,如果要调用method1(),你必须把这个调用放在try/catch程序块当中,因为它会抛出Checked exception。

!XN|*TZp7O&S\o:D051Testing软件测试网V)P.K Ex/go1wU

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

i cG8|$E1tX3A0

3K9YI*jOS`6d.O0  现在,让我们再来看看method3()。它调用了method1()却没有把这个调用放在try/catch程序块当中。它是通过声明它会抛出method1()会抛出的exception来避免这样做的。它没有捕获这个exception,而是把它传递下去。实际上main()方法也可以这样做,通过声明它会抛出Checked exception来避免使用try/catch程序块(当然我们反对这种做法)。51Testing软件测试网 F7KUWW^!Q&y

51Testing软件测试网8v,jS:P,M*T#Z

  小结一下:51Testing软件测试网a5w nYYDl#q

51Testing软件测试网4\|2HS0qV G4y+L

  * Runtime exceptions:

CbtkVLM0

+yGyB5I)mA6k0  在定义方法时不需要声明会抛出runtime exception;

qM5t D/GC1Z!F051Testing软件测试网}i8f/dH%Ov

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

*f.Wbg?C |0

p6j TyqO'j9u/yV:{0  runtime exception是从java.lang.RuntimeException或java.lang.Error类衍生出来的。

DW }'B{W.R L0

I a2E`$r7Dv3`{n0  * Checked exceptions:

W4AU,M B051Testing软件测试网9x+p\/XH3],I2D?0f

  定义方法时必须声明所有可能会抛出的checked exception;51Testing软件测试网!rY(s:[$J&},} F:YJ;X

g`[@'v&d/Z(I0  在调用这个方法时,必须捕获它的checked exception,不然就得把它的exception传递下去;

d;K2vSCK(N/N051Testing软件测试网B|l[ k

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

i,wI(c \0r5R0

W3|1BF9Me0  二、逻辑上

\)F*_+n?;bBt051Testing软件测试网htS"QK%wv

  从逻辑的角度来说,checked exceptions和runtime exception是有不同的使用目的的。checked exception用来指示一种调用方能够直接处理的异常情况。而runtime exception则用来指示一种调用方本身无法处理或恢复的程序错误。51Testing软件测试网`T.q,Cj+w

51Testing软件测试网([2[x^g

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

H!w2ra wHud0

nsU Q @ b Q]0  再看下面这个例子:51Testing软件测试网y7KyMLv _?

oE$[.w Cp`[T0  public void method()51Testing软件测试网 o f(lh4F n:Ro
  {
X&N.H:r Kt1GJ0  int [] numbers = { 1, 2, 3 };51Testing软件测试网s`L;Nvk^t
  int sum = numbers[0] numbers[3];
%H$p)Q$k2d$N0  }

j3n1}|i&^7V-w0

[CI@@(xQ'GO0  在运行方法method()时会遇到ArrayIndexOutOfBoundsException(因为数组numbers的成员是从0到2)。对于这个异常,调用方无法处理/纠正。这个方法method()和上面的method2()一样,都是runtime exception的情形。上面我已经提到,runtime exception用来指示一种调用方本身无法处理/恢复的程序错误。而程序错误通常是无法在运行过程中处理的,必须改正程序代码。51Testing软件测试网!L1n5_.C^j$Azl0|

51Testing软件测试网Qb s9L8DAHr%T-F

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

51Testing软件测试网7Cz;]R5d$I#ZAC

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


TAG: JAVA 异常处理 编程基础

 

评分:0

我来说两句

Open Toolbar