Java语言中两种异常的差别

Java提供了两类主要的异常:Runtime ExceptionChecked Exception

所有的Checked Exception是从java.lang.Exception类衍生出来的,
而Runtime Exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。


它们的不同之处表现在两方面:机制上和逻辑上。

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

请看下面CheckedException的定义:

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 e) {
		e.printStackTrace();
	}
	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 Exception和Runtime Exception是有不同的使用目的的。


Checked Exception用来指示一种调用方能够直接处理的异常情况。
而Runtime Exception则用来指示一种调用方本身无法处理或恢复的程序错误。

 

Checked Exception迫使你捕获它并处理这种异常情况。


以java.net.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程序块吗?

 [文章均为转载加整理修改,方便自己学习用,由于收藏时部分文章没版权声明,因此未注明出处,若侵权请指出]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值