总结下Java中的异常处理,基本来自Java编程思想和真实的归宿的博文,如有错误或者不足,欢迎交流讨论。
异常(Exception)即程序的意外情况,C及早期的语言的错误处理往往是预定俗成的,并不是语言的一部分,语言本身不强制程序员检查和处理异常,如果每次调用方法都要检查错误,代码将变得难以阅读,且开发的效率无疑要大大降低,出错后要检查特定的错误,并在多处处理,这对于检查和处理都很不方便,对于构造大型、健壮、可维护的程序而言,这种错误处理方式显然不适合。对于有异常处理的语言而言,异常处理是程序强制程序员检查并处理异常的,异常的处理方法有两种,要么恢复到正常情况,要么终止程序并给出错误的描述(异常所在的函数、异常类型等)以帮助程序员检查错误。
Java中所有的异常类均继承自Throwable类(java.lang.Throwable,可参考HTML格式的java文档),内置的异常类大部分都在java.lang包里,还有一些在util、io、net等包里。如果Java内置异常不能满足用户的需求,可以自定义异常,即创建自己的异常类。Throwable包含两个重要子类,Exception和Error。
1. Exception
表示是程序本身可以处理的异常,有两个分支,RuntimeException和IOException,前者是程序错误导致的异常,后者是程序本身没有错误,是由IO这类问题导致的异常。RuntimeException及其子类表示“JVM常用操作”引发的错误,进行Java程序设计应该重点关注RuntimeException,出现RuntimeException一定是程序员引发的。
2. Error
Erro(错误)是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与程序员无关,而表示代码运行时JVM出现的问题。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止。
异常处理有两种基本模型:终止模型,这种错误非常关键,以至程序无法返回到异常发生的地方继续执行,异常一旦抛出,就表明错误无法挽回;恢复模型,希望遇见错误时就不能抛出异常,而是调用方法来修正错误,或者,把try块放在while循环里(循环设为永真,若恢复,则break),不断进入try块,直到满意的结果。
当一个方法内出现异常时,方法内又没有捕获异常的代码,那么就必须将异常抛出。在方法形列表后使用throws关键字给出潜在的异常类型列表(异常说明),在方法内使用throw关键字抛出异常,但是有异常说明不一定就要抛出异常,在定义抽象类和接口时很有用。
package exception;
/** * 抛出异常 * @author 小锅巴 * @date 2016年3月31日下午4:17:28 * http://blog.csdn.net/xiaoguobaf */
class MyException extends Exception{
public MyException(){}
public MyException(String msg){
super(msg);
}
//再次说明,在一个.class文件中可以有两个main方法,运行时选择就是,这里还是为了方便测试
public static void main(String[] args) throws Exception{//有异常声明可以不抛出异常
System.out.println("throw nothing !");
}
}
class MyException01 extends Exception{
}
public class FullConstructor {
public static void f() throws MyException{
System.out.println("Throwing MyException form f()");
throw new MyException();
}
public static void g() throws MyException{
System.out.println("Throwing MyException form g()");
throw new MyException("Originated in g()");
}
//可以在异常列表里声明将抛出多个类型的异常,但是实际上只能抛出一个,因为抛出一个程序就终止了,
public static void main(String[] args) throws MyException, MyException01{
throw new MyException01();//抛出异常
// System.out.println("Test");//抛出异常后程序就终止了,属于终止模型,eclipse提示:Unreachable code
}
}
/** 输出: 第一个main函数:throw nothing ! 第二个main函数:Exception in thread "main" exception.MyException01 at exception.FullConstructor.main(FullConstructor.java:31) */
特例是RuntimeException,不用在异常说明里给出异常类型列表,其输出被报告给了System.err,并在方法退出前调用printStackTrace()
package exception;
/** * 特例,RuntimeException不需要异常说明 * @author 小锅巴 * @date 2016年4月1日下午2:10:10 * http://blog.csdn.net/xiaoguobaf */
public class TestRuntimeException {
static void f(){
throw new RuntimeException("From f()");
}
static void g(){
f();
}
public static void main(String[] args) {
g();
}
}
/** * 输出: Exception in thread "main" java.lang.RuntimeException: From f() at exception.TestRuntimeException.f(TestRuntimeException.java:12) at exception.TestRuntimeException.g(TestRuntimeException.java:15) at exception.TestRuntimeException.main(TestRuntimeException.java:18) */
监控区域(guarded region):一段能产生异常的代码,一般包含在try块中,在try块中的异常即被捕获了,try中可以“尝试”各种方法调用(可能会产生异常),在try块中发生异常程序将不会结束;try块后面跟着处理异常的代码(异常处理程序),一般包含在catch块中(如果没有catch块,则必须要有finally块)。
一个try捕获多个异常类型,并对不同类型的异常做出不同的处理,一旦和其中的任何一个catch相匹配,则catch块执行结束后即视为异常处理完,不再执行后面的catch块,这与switch case不同。
package exception;
/** * 捕获多个异常 * @author 小锅巴 * @date 2016年4月1日下午2:37:19 * http://blog.csdn.net/xiaoguobaf */
class Exception01 extends Exception{
}
class Exception02 extends Exception{
}
class Exception03 extends Exception{
}
public class CatchMultipleException {
public static void main(String[] args) throws Exception01,Exception02,Exception03{
int i = 0;
while(true){//恢复模型,不停try,直到不出现异常
i++;
try{
if(i == 1)
throw new Exception01();
else if(i == 2)
throw new Exception02();
else if(i == 3)
throw new Exception03();
}catch(Exception01 e1){
System.out.println("Throw Exception01");
}catch(Exception02 e2){
System.out.println("Throw Exception02");
}catch(Exception03 e3){
System.out.println("Throw Exception03");
}
if(i == 4){
System.out.println("No exceptions, quit loop!");
break;
}
}
}
}
/** * 输出: Throw Exception01 Throw Exception02 Throw Exception03 No exceptions, quit loop! */
捕获异常时,虽然catch块中不一定用到异常类型,但catch中的异常类型不可缺省。此外还可捕获所有异常,在一个catch块中捕获多个异常,那么显然类型是多个异常的基类类型,不过要放在异常处理程序的最后面,否则他后面的对到导出类型的异常处理程序将被忽略。
finally的使用:不论异常是否抛出,finally字句总能被执行;就算有return在finally之前,finally子句还是会得到执行;涉及break和continue,finally子句还是会得到执行。一般情况下不论try块里面发生什么,finally子句都会得到执行。当要把除内存之外的资源恢复到它们的初始状态时,就需要用到finally块,如文件、IO等。
以下情况下finally中的子句不会执行:
(1)在finally语句块中发生了异常。
(2)在前面的代码中用了System.exit()退出程序。
(3)程序所在的线程死亡。
(4)关闭CPU。
package exception;
/** * finally子句 * 说明:Exception01,Exception02,Exception03已经在CatchMultipleException.java中定义了,他们在同一个包下 * @author 小锅巴 * @date 2016年4月1日下午5:04:30 * http://blog.csdn.net/xiaoguobaf */
public class Finally {
public static void main(String[] args) throws Exception01,Exception02,Exception03{
int i = 0;
while(true){//恢复模型,不停try,直到不出现异常
i++;
try{
if(i == 1)
throw new Exception01();
else if(i == 2)
throw new Exception02();
else if(i == 3)
throw new Exception03();
else if(i == 4){
System.out.println("No exception, continue!");
continue;
}
else if(i == 5){
// System.out.println("No exception, break !");
// break;
System.out.println("No exception, return !");
return;
}
}catch(Exception01 e1){
System.out.println("Throw Exception01");
}catch(Exception02 e2){
System.out.println("Throw Exception02");
}catch(Exception03 e3){
System.out.println("Throw Exception03");
}
finally{
System.out.println("In finally clause");
}
}
}
}
/** * 输出: Throw Exception01 In finally clause Throw Exception02 In finally clause Throw Exception03 In finally clause No exception, continue! In finally clause No exception, break ! In finally clause 将break相关的注释掉,执行return,则输出: Throw Exception01 In finally clause Throw Exception02 In finally clause Throw Exception03 In finally clause No exception, continue! In finally clause No exception, return ! In finally clause */
先说下栈轨迹(stack trace):printStackTrace(),该方法返回一个由栈轨迹中的元素所构成的数组,第一个数组元素即是栈顶元素,是方法调用序列中的最后一个方法调用,最后一个数组元素是栈底元素,是方法调用的第一个方法调用。其中的元素可由getStackTrace()方法来访问。
重新抛出异常:捕获异常后再重新抛出异常,会把异常抛给上一级环境中的异常处理程序,此时同一个try块后面的catch子句将被忽略,若只是把当前异常对象重新抛出,那么printStackTrace()显示的还是原来异常抛出点的调用栈信息,而非重新抛出点的信息,想要更新抛出的调用栈信息,可调用fillInStackTrace()方法,它返回Throwable对象
package exception;
/** * 重新抛出异常 * @author 小锅巴 * @date 2016年4月1日下午7:43:23 * http://blog.csdn.net/xiaoguobaf */
public class Rethrowing {
public static void f() throws Exception{
throw new Exception("thrown form f()");//异常参数,在异常构造中
}
public static void g() throws Exception{
try {
f();
} catch (Exception e) {
System.out.println("Inside g(),e.printStackTrace()");
e.printStackTrace(System.out);//异常信息会被发送给System.out输出流
throw e;//重新抛出异常,没有使用fillInStackTrace(),故还是原抛出点的异常信息
}
}
public static void h() throws Exception{
try {
f();
} catch (Exception e) {
System.out.println("Inside h(),e.printStackTrace()");
e.printStackTrace(System.out);
//使用fillInStackTrace(),覆盖了原抛出点的异常信息,
throw (Exception)e.fillInStackTrace();//fillInStackTrace()返回的是Throwable类型的异常,与异常说明类型不相符,所以要向下转型
}
}
public static void main(String[] args) {
try{
g();
}catch(Exception e){
System.out.println("main g(): printStackTrace()");
e.printStackTrace(System.out);
}
try {
h();
} catch (Exception e) {
System.out.println("main h(): printStackTrace()");
e.printStackTrace(System.out);
}
}
}
/** 输出: Inside g(),e.printStackTrace() java.lang.Exception: thrown form f() at exception.Rethrowing.f(Rethrowing.java:10) at exception.Rethrowing.g(Rethrowing.java:14) at exception.Rethrowing.main(Rethrowing.java:33) main g(): printStackTrace() java.lang.Exception: thrown form f() at exception.Rethrowing.f(Rethrowing.java:10) at exception.Rethrowing.g(Rethrowing.java:14) at exception.Rethrowing.main(Rethrowing.java:33) Inside h(),e.printStackTrace() java.lang.Exception: thrown form f() at exception.Rethrowing.f(Rethrowing.java:10) at exception.Rethrowing.h(Rethrowing.java:23) at exception.Rethrowing.main(Rethrowing.java:39) main h(): printStackTrace() java.lang.Exception: thrown form f() at exception.Rethrowing.h(Rethrowing.java:28) at exception.Rethrowing.main(Rethrowing.java:39) */
异常链:若捕获异常之后抛出另外一种异常,原来异常发生点的信息会丢失;若要在捕获一个异常后抛出另一异常,并且希望把原来异常的信息保存下来。只有Error、Exception、RuntimeException这三种基本的异常类(不包括子类)提供了带cause的构造器,其他类型的异常若要链接,应该使用initCause()方法。
package exception;
/** * 异常链 * @author 小锅巴 * @date 2016年4月1日下午7:59:19 * http://blog.csdn.net/xiaoguobaf */
class ExceptionChain extends Exception{
public static void f() throws Exception{
throw new Exception();
}
}
public class ChainingExceptions {
public static void main(String[] args) throws Exception{
try {
ExceptionChain ec = new ExceptionChain();
ec.f();
} catch (Exception e) {
e.initCause(new NullPointerException());//iniCause()方法参数里面的异常的类型即是重新抛出后的异常类型
throw e;
}
}
}
/** 输出: Exception in thread "main" java.lang.Exception at exception.ExceptionChain.f(ChainingExceptions.java:10) at exception.ChainingExceptions.main(ChainingExceptions.java:17) Caused by: java.lang.NullPointerException at exception.ChainingExceptions.main(ChainingExceptions.java:19) */
在Java中提供了一些异常用来描述经常发生的错误,对于这些异常,有的需要程序员进行捕获处理或声明抛出,有的是由Java虚拟机自动进行捕获处理。Java中常见的异常类:
runtimeException子类:
1、 java.lang.ArrayIndexOutOfBoundsException
数组索引越界异常。当对数组的索引值为负数或大于等于数组大小时抛出。
2、java.lang.ArithmeticException
算术条件异常。譬如:整数除零等。
3、java.lang.NullPointerException
空指针异常。当应用试图在要求使用对象的地方使用了null时,抛出该异常。譬如:调用null对象的实例方法、访问null对象的属性、计算null对象的长度、使用throw语句抛出null等等
4、java.lang.ClassNotFoundException
找不到类异常。当应用试图根据字符串形式的类名构造类,而在遍历CLASSPAH之后找不到对应名称的class文件时,抛出该异常。
5、java.lang.NegativeArraySizeException 数组长度为负异常
6、java.lang.ArrayStoreException 数组中包含不兼容的值抛出的异常
7、java.lang.SecurityException 安全性异常
8、java.lang.IllegalArgumentException 非法参数异常
IOException
IOException:操作输入流和输出流时可能出现的异常。
EOFException 文件已结束异常
FileNotFoundException 文件未找到异常
其他
ClassCastException 类型转换异常类
ArrayStoreException 数组中包含不兼容的值抛出的异常
SQLException 操作数据库异常类
NoSuchFieldException 字段未找到异常
NoSuchMethodException 方法未找到抛出的异常
NumberFormatException 字符串转换为数字抛出的异常
StringIndexOutOfBoundsException 字符串索引超出范围抛出的异常
IllegalAccessException 不允许访问某类异常
InstantiationException 当应用程序试图使用Class类中的newInstance()方法创建一个类的实例,而指定的类对象无法被实例化时,抛出该异常
若不恰当地finally子句,将导致异常被忽略,如:
package exception;
import javax.management.RuntimeErrorException;
/** * 异常的丢失 * @author 小锅巴 * @date 2016年4月1日下午8:26:53 * http://blog.csdn.net/xiaoguobaf */
class VeryImportantException extends Exception{
public String toString(){
return "A verry important exception!";
}
}
class HoHumException extends Exception{
public String toString(){
return "A trivale exception";
}
//测试在finally中使用return
public static void main(String[] args){
try {
throw new RuntimeException();
}finally{
return;//finally中没有处理抛出异常,而是返回,执行该main函数什么也不会输出
}
}
}
public class LostException {
void f() throws VeryImportantException{
throw new VeryImportantException();
}
void g() throws HoHumException {
throw new HoHumException();
}
public static void main(String[] args) {
try {
LostException le = new LostException();
try{//异常监控区域的嵌套
le.f();
}finally{//没有catch块,使用finally子句
le.g();//不恰当地使用finally子句,导致下面的catch块不会执行,VeryImportantException被忽略
}
} catch (Exception e) {
System.out.println(e);
}
}
}
/** 输出: A trivale exception */
因为面向对象中的继承,将父类使用的代码应用到其子类时,是能够正常工作的,正是因为此,当一个类继承子某个异常类时,子类中处理的异常不会像继承一样增加信息,而是不能增加(增加能处理的异常类),因为如果增加了,那么在父类中本来不会抛出异常,能正常工作,将父类中该代码应用到子类时,因为增加,抛出了异常,该代码就不能工作了,这违背了继承代码复用的原则。尤其是一个类实现一个接口并继承自一个类,父类和接口有同名方法,该方法异常的抛出要谨慎。具体例子就不写了,java编程思想271页有。
前面说了一点,即捕获所有异常,子类的对象也可以匹配父类的处理程序,还是继承的代码复用原则,能处理父类的,子类肯定也能处理,但注意要放在最后,否则其后面的子类的处理程序将被忽略。
若构造器中抛出了异常,清理行为也许就不能正常工作,因为在构造器内部对象还没有成功创建。
对于构造阶段可能抛出异常并要求清理的类(应该是文件、IO之类的,毕竟是资源,否则浪费了),最安全的方式是使用嵌套的try子句,这里直接贴java编程思想的图吧,因为作者貌似使用了自己的库。
我只想说一点,创建完对象立马就清理掉,那干嘛要创建呢?所以,我的理解是,如果有构造器中可能发生异常并又必须清理(资源),那么加一个标志位,通过判断标志位可以在成功创建时使用该对象,然后调用dispose清理对象,这样就可以确保创建成功时使用对象并清理,不成功时直接清理。
参考资料:
1. Java编程思想
2. 真实的归宿的博文
3. Java核心技术