第9章异常处理了解异常处理的基础知识理解异常处理机制自定义异常类的使用9.1异常处理基础异常就是在程序的运行过程中所发生的异常事件,它中断指令的正常执行。Java中提供了一种独特的处理异常的机制,通过异常来处理程序设计中出现的错误。当程序运行出现异常时,Java运行环境就用异常类Exception的相应子类创建一个异常对象,并等待处理,例如,读取一个不存在的文件时,运行环境就用异常类IOException创建一个对象。异常对象可以调用如下方法得到或输出有关异常的信息。9.1.1异常示例【例9-1】读取一个不存在的文件时,系统编译的时候报异常。importjava.io.*;classExceptionDemo1{publicstaticvoidmain(Stringargs[])throwsIOException{FileInputStreamfis=newFileInputStream(test.txt);intb;while((b=fis.read())!=-1){System.out.print(b);}fis.close();}}查看程序的编译结果:编译结果提示:第4行会出现异常,必须对其进行捕获或声明抛出,否则程序编译无法通过。9.1.2异常类的层次在jdk中,每个包中都定义了异常类,而所有的异常类都直接或间接地继承于Throwable类。图9-1为jdk中异常类的继承关系。异常的继承结构:基类为Throwable,Error和Exception继承Throwable,RuntimeException和IOException等继承Exception,具体的RuntimeException继承RuntimeException。9.1.3异常类的分类Java中的异常类可分为两大类:Error和Exception。Error——动态链接失败,虚拟机错误等,通常Java程序不应该捕获这类异常,也不会抛弃这种异常。Exception——包括运行时异常和非运行时异常。1)运行时异常:继承于RuntimeException的类都属于运行时异常,例如算术异常(除零错)、数组下标越界异常等等。由于这些异常产生的位置是未知的,Java编译器允许程序员在程序中不对它们做出处理。2)非运行时异常:除了运行时异常之外的其他由Exception继承来的异常类都是非运行时的异常,例如FileNotFoundException(文件未找到异常)。Java编译器要求在程序中必须处理这种异常,捕获异常或者声明抛弃异常。1.Error体系类型异常的特点Error类体系描述了Java运行系统中的内部错误以及资源耗尽的情形。应用程序不应该抛出这种类型的对象(一般是由虚拟机抛出)。如果出现这种错误,除了尽力使程序安全退出外,在其他方面是无能为力的。所以,在进行程序设计时,应该更关注Exception体系。2.Exception体系类型异常的特点Exception体系包括RuntimeException体系和其他非RuntimeException的体系(1)RuntimeExceptionRuntimeException体系包括错误的类型转换、数组越界访问和试图访问空指针等等。处理RuntimeException的原则是:如果出现RuntimeException,那么一定是程序员的错误。例如,可以通过检查数组下标和数组边界来避免数组越界访问异常。(2)其他(IOException等等)这类异常一般是外部错误,例如试图从文件尾后读取数据等,这并不是程序本身的错误,而是在应用环境中出现的外部错误。9.2异常的处理Java语言中有两种异常处理机制:捕获异常和声明抛弃异常。(1)捕获异常:当Java运行环境得到一个异常对象时,它将会沿着方法的调用栈逐层回溯,寻找处理这一异常的代码。找到能够处理这种类型的异常的方法后,运行环境把当前异常对象交给这个方法进行处理,这一过程称为捕获(catch)异常。这是积极的异常处理机制。如果Java运行环境找不到可以捕获异常的方法,则运行环境将终止,相应的Java程序也将退出。(2)声明抛弃异常:如果一个方法并不知道如何处理所出现的异常,则可在方法声明时,声明抛弃(throws)异常。这是一种消极的异常处理机制。9.2.1捕获异常捕获异常是通过try-catch-finally语句实现的。try{......}catch(ExceptionName1e){......}catch(ExceptionName2e){......}......}finally{......}◇try捕获异常的第一步是用try{…}选定捕获异常的范围,由try所限定的代码块中的语句在执行过程中可能会生成异常对象并抛弃。◇catch每个try代码块可以伴随一个或多个catch语句,用于处理try代码块中所生成的异常事件。catch语句只需要一个形式参数指明它所能够捕获的异常类型,这个类必须是Throwable的子类,运行时系统通过参数值把被抛弃的异常对象传递给catch块。在catch块中是对异常对象进行处理的代码,与访问其它对象一样,可以访问一个异常对象的变量或调用它的方法。getMessage()是类Throwable所提供的方法,用来得到有关异常事件的信息,类Throwable还提供了方法printStackTrace()用来跟踪异常事件发生时执行堆栈的内容。例如:try{......}catch(FileNotFoundExceptione){System.out.println(e);System.out.println(message:+e.getMessage());e.printStackTrace(System.out);}catch(IOExceptione){System.out.println(e);}catch语句的顺序:捕获异常的顺序和catch语句的顺序有关,当捕获到一个异常时,剩下的catch语句就不再进行匹配。因此,在安排catch语句的顺序时,首先应该捕获最特殊的异常,然后再逐渐一般化。也就是一般先安排子类,再安排父类。◇finally捕获异常的最后一步是通过finally语句为异常处理提供一个统一的出口,使得在控制流转到程序的其它部分以前,能够对程序的状态作统一的管理。不论在try代码块中是否发生了异常事件,finally块中的语句都会被执行。9.2.2声明抛弃异常1.声明抛弃异常如果在一个方法中生成了一个异常,但是这一方法并不确切地知道该如何对这一异常事件进行处理,这时,一个方法就应该声明抛弃异常,使得异常对象可以从调用栈向后传播,直到有合适的方法捕获它为止。2.抛出异常抛出异常就是产生异常对象的过程,首先要生成异常对象,异常或者由虚拟机生成,或者由某些类的实例生成,也可以在程序中生成。在方法中,抛出异常对象是通过throw语句实现的。9.3自定义异常类当Java内置的异常都不能明确的说明异常情况的时候,开发人员往往需要定义一些异常类用于描述自身程序中的异常信息,以区分其他程序的异常信息。需要注意的是,唯一有用的就是类型名这个信息,所以不要在异常类的设计上花费精力。实现自定义异常类的方法如下:类java.lang.Throwable是所有异常类的基类,它包括两个子类:Exception和Error,Exception类用于描述程序能够捕获的异常,如ClassNotFoundException。Error类用于指示合理的应用程序不应该试图捕获的严重问题,如虚拟机错误VirtualMachineError自定义异常类必须是Throwable的直接或间接子类。自定义异常类可以继承Throwable类或者Exception,而不要继承Error类。自定义异常类之间也可以有继承关系需要为自定义异常类设计构造方法,以方便构造自定义异常对象。注意:一个方法所声明抛弃的异常是作为这个方法与外界交互的一部分而存在的。所以,方法的调用者必须了解这些异常,并确定如何正确的处理他们。9.3.1继承Exception类【例9-3】继承Exception类。publicclassMyFirstExceptionextendsException{publicMyFirstException(){super();}publicMyFirstException(Stringmsg){super(msg);}publicMyFirstException(Stringmsg,Throwablecause){super(msg,cause);}publicMyFirstException(Throwablecause){super(cause);}}//自定义异常类的主要作用是区分异常发生的位置,当用户遇到异常时,根据异常名就可以知道哪里有异常,根据异常提示信息进行修改。9.3.2继承Throwable类【例9-4】继承Throwable类。publicclassMySecondExceptionextendsThrowable{publicMySecondException(){super();}publicMySecondException(Stringmsg){super(msg);}publicMySecondException(Stringmsg,Throwablecause){super(msg,cause);}publicMySecondException(Throwablecause){super(cause);}}9.3.3自定义异常类的使用【例9-5】自定义异常类的使用。publicclassTestMyException{publicstaticvoidfirstException()throwsMyFirstException{thrownewMyFirstException(firstException()methodoccursanexception!);}publicstaticvoidsecondException()throwsMySecondException{thrownewMySecondException(secondException()methodoccursanexception!);}publicstaticvoidmain(String[]args){try{TestMyException.firstException();TestMyException.secondException();}catch(MyFirstExceptione1){System.out.println(Exception:+e1.getMessage());e1.printStackTrace();}catch(MySecondExceptione2){System.out.println(Exception:+e2.getMessage());e2.printStackTrace();}}}程序运行的结果如下所示:当一个try块后面跟着多个catch块时,如果发生的异常匹配第一个catch块的参数,便将异常处理权利交给第一个catch块。如果发生的异常与第一个catch块不匹配,便看是否与第二个catch块匹配,依次下去,如果到最后依然无法匹配该异常,便需要在方法声明中添加一条throw语句,将该异常抛出。9.4练习思考题1、为了捕获一个异常,代码必须放在下面()语句块中。A、try块B、catch块C、throws块D、finally块2、下列常见的系统定义的异常中,有可能是网络原因导致的异常是()。A、ClassNotFoundExceptionB、IOExceptionC、FileNotFoundExceptionD、UnknownHostException3、在代码中,使