Java异常处理的基础知识Word下载.docx
- 文档编号:16362516
- 上传时间:2022-11-23
- 格式:DOCX
- 页数:14
- 大小:207.88KB
Java异常处理的基础知识Word下载.docx
《Java异常处理的基础知识Word下载.docx》由会员分享,可在线阅读,更多相关《Java异常处理的基础知识Word下载.docx(14页珍藏版)》请在冰豆网上搜索。
异常抛出后,运行时系统从生成对象的代码开始,沿方法的调用栈逐层回溯查找,直到包含相应处理的方法,并把异常对象交给该方法为止,这个过程称为捕获(catch)一个异常。
简单地说,发现异常的代码可以“抛出”一个异常,运行系统“捕获”该异常,交由程序员编写的相应代码进行异常处理。
3.异常处理的类层次
Java通过错误类(Error)和异常类(Exception)来处理错误和异常,而它们都是Throwable类的子类,分别用来处理两组异常。
它们的层次结构如图9-1所示。
图9-1异常类层次结构
注意:
Throwable类是直接由Object类继承而来的一个类,可见Java对异常控制是非常重视的。
4.程序对错误与异常的三种处理方式:
⑴程序不能处理的错误
Error类为错误类,如内存溢出、栈溢出等。
这类错误一般由系统进行处理,程序本身无需捕获和处理。
例如,运行没有main方法的类将产生NoClassDefFoundError错误。
⑵程序应避免而不捕获的异常
对于运行时异常类(RuntimeException),如数组越界等,在程序设计正常时不会发生,在编程时使用数组长度a.length来控制数组的上界即可避免异常发生,而无须使用try-catch-finally语句。
因此,这类异常应通过程序调试尽量避免而不是去捕获它。
⑶必须捕获的异常
有些异常在编写程序时是无法预料了,如文件没找到异常、网络通信失败异常等。
因此,为了保证程序的健壮性,Java要求必须对可能出现这些异常的代码使用try-catch-finally语句,否则编译无法通过。
【例9-1】文件没有找到异常类。
设计思路:
本例访问文件autoexec.bat。
在程序中使用了FileInputStream类,在访问文件时会产生文件不存在的异常对象(FileNotFoundException),所以必须捕获的这个异常,否则编译就会出错。
代码:
importjava.io.*;
publicclassTry3
{
publicstaticvoidmain(Stringargs[])
{
FileInputStreamfis=newFileInputStream("
autoexec.bat"
);
System.out.println("
Icannotfoundthisfile!
"
}
}
程序运行效果如下:
图9-1
5.常见的公用异常类
下面介绍常见的异常类,它们都是RuntimeException的子类。
⑴算术异常ArithmeticException
如果除数为除0,或用0取模会产生ArithmeticException,其它算术操作不会产生异常。
⑵空指针异常NullPointerException
当程序试图访问一个空对象中的变量或方法,或一个空数组中的元素时则会引发NullPointerException异常。
例如,
inta[]=null;
a[0]=0;
//访问长度为0的数组,产生NullPointerException
Stringstr=null;
System.out.println(str.length());
//访问空字符串的方法,产生NullPointerException
⑶类型强制转换异常ClassCastException
进行类型强制转换时,对于不能进行的转换操作产生ClassCastException异常。
Objectobj=newObject();
Stringstr=(String)obj;
上述语句试图把Object对象强制转换成String对象str,而obj既不是String的实例,也不是String子类的实例,系统不能转换时产生ClassCastException异常。
⑷数组负下标异常NegativeArraySizeException
如果一个数组的长度是负数,则会引发NegativeArraySizeException异常。
inta[]=newint[-1];
//产生NegativeArraySizeException异常
⑸数组下标越界异常ArrayIndexOutOfBoundsException
试图访问数组中的一个非法元素时,引发ArrayIndexOutOfBoundsExceptiony异常。
inta[]=newint[1];
a[1]=1;
//数组下标越界异常
9.2异常类的产生、捕获和处理
异常处理的理论似乎十分繁琐,实际使用时却并不复杂。
下面我们先通过一个例子来看一下异常产生到捕获并处理的过程。
9.2.1异常的产生
【例9-2】产生数组下标越界异常和除数为0异常。
打印一个数组的所有值。
程序编译时没有问题,但运行时正常输出了循环的前4句,但在试图输出A[4]时,Java抛出了一个数组越界异常类(java.lang.ArrayIndexOutOfBoundsException),以及异常发生所在的方法(Try1.main),同时终止程序运行;
publicclassTry1
inti=0;
inta[]={5,6,7,8};
for(i=0;
i<
5;
i++)
a["
+i+"
]="
+a[i]);
图9-2
9.2.2使用try-catch-finally语句捕获和处理异常
一般来说,系统捕获抛出的异常对象并输出相应的信息,同时终止程序运行,导致其后程序无法运行。
这其实并不是人们所期望的,因此就需要能让程序来接收和处理异常对象,从而不会影响其他语句的执行,这就是捕获异常的意义所在。
在Java的异常处理机制中,提供了try-catch-finally语句来捕获和处理一个或多个异常,语法格式如下:
try
<
语句1>
catch(ExceptionType1e)
语句2>
finally
语句3>
其中,<
是可能产生异常的代码;
<
是捕获某种异常对象时进行处理的代码,ExceptionType1代表某种异常类,e为相应的对象;
是其后必须执行的代码,无论是否捕获到异常。
catch语句可以有一个或多个,但至少要有一个catch语句,finally语句可以省略。
try-catch-finally语句的作用是,当try语句中的代码产生异常时,根据异常的不同,由不同catch语句中的代码对异常进行捕获并处理;
如果没有异常,则catch语句不执行;
而无论是否捕获到异常都必须执行finally中的代码。
【例9-3】异常捕获和处理。
本例使用try-catch-finally语句对例9-2中产生的异常进行捕获和处理。
publicclassTry2{
publicstaticvoidmain(Stringargs[]){
i++){
try{
catch(ArrayIndexOutOfBoundsExceptione){
数组下标越界异常!
finally{
fianllyi="
+i);
图9-3
通过这个例子我们再来深入讨论try-catch-finally语句,以及使用时要注意的问题。
⑴try语句
try语句大括号{}中的这段代码可能会抛出一个或多个异常。
也就是说,当某段代码在运行时可能产生异常的话,需要使用try语句来试图捕获这个异常
⑵catch语句
catch语句的参数类似于方法的声明,包括一个异常类型和一个异常对象。
catch语句可以有多个,分别处理不同类的异常。
Java运行时系统从上向下分别对每个catch语句处理的异常类型进行检测,直到找到与类型相匹配的catch语句为止。
如果程序产生的异常和所有catch的处理的异常都不匹配,则这个异常将由Java虚拟机捕获并处理,此时与不使用try-catch-finally语句是一样的,这显然也不是我们所期望的结果。
因此一般在使用catch语句时,最后一个将捕获Exception这个所有异常的超类,从而保证异常由对象自身来捕获和处理。
⑶finally语句
try所限定的代码中,当抛出一个异常时,其后的代码不会被执行。
通过finally语句可以指定一块代码,无论try所指定的程序块中抛出异常,也无论catch语句的异常类型是否与所抛出的异常的类型一致,finally所指定的代码都要被执行,它提供了统一的出口。
该语句是可以省略的。
9.3抛出异常
如前所述,在捕获一个异常前,必须有一段代码生成一个异常对象并把它抛出。
抛出异常的既可以是Java运行时系统,如例9-2;
也可以是程序员自己编写的代码,即在try语句中的代码本身不会有系统产生异常,而是由程序员故意抛出异常。
9.3.1使用throw语句抛出异常
使用throw语句抛出异常格式如下:
throw<
异常对象>
其中,throw是关键字,<
是创建的异常类对象。
【例9-4】抛出异常。
本例为求1-20的阶乘。
在该例中使用主动抛出异常、再捕获并处理异常的方式解决数据溢出的问题。
在每次乘法前先判断,如果结果会溢出,则由throw语句抛出一个异常,再由catch语句对捕获的异常进行处理。
publicclassTry5{
publicvoidrun(bytek){
bytey=1,i=1;
System.out.print(k+"
!
="
//不换行输出
for(i=1;
=k;
try
if(y>
Byte.MAX_VALUE/i)//Integer类的常量,表示最大值
thrownewException("
overflow"
//溢出时抛出异常
else
y=(byte)(y*i);
catch(Exceptione)
exception:
+e.getMessage());
e.printStackTrace();
//显示异常信息
System.exit(0);
System.out.println(y);
Try5a=newTry5();
for(bytei=1;
10;
a.run(i);
图9-4
9.3.2抛出异常的方法与调用方法处理异常
⑴抛出异常的方法
在方法声明中,添加throws子句表示该方法将抛出异常。
带有throws子句的方法的声明格式如下:
[<
修饰符>
]<
返回值类型>
方法名>
([<
参数列表>
])[throws<
异常类>
]
其中,throws是关键字,<
是方法要抛出的异常类,可以声明多个异常类,用逗号隔开。
将throws子句与throws在语法和使用上要加以区别。
⑵由调用方法处理异常
由一个方法抛出异常后,系统将异常向上传播,由调用它的方法来处理这些异常。
【例9-5】抛出异常的方法与调用方法处理异常
本例在计算阶乘的calc方法中抛出数据溢出的异常。
程序运行时,在Calc方法中生成的异常通过调用栈传递给run方法,由run方法进行处理。
publicclassTry6{
publicvoidcalc(bytek)throwsException{//抛出异常
i++){
try{
catch(Exceptione){
publicvoidrun(bytek){//捕获并处理异常
calc(k);
catch(Exceptione){
publicstaticvoidmain(Stringargs[]){
Try6a=newTry6();
}程序运行效果如下:
图9-5
同throw一样,如果某个方法声明抛出异常,则调用它的方法必须捕获及处理异常,否则会出现异常错误。
9.3.3由方法抛出异常交系统处理
对于程序中需要处理的异常,一般编写try-catch-finally语句捕获并处理;
而对于程序中无法处理必须由系统处理的异常,可以使用throw语句在方法中抛出异常交系统处理。
例如,对于文件流操作,将必须捕获的系统定义的异常交由系统系统处理。
publicclassDemo1//异常用法举例
staticinta,b,c;
try
{a=100;
b=Integer.parseInt(args[0]);
if(b==13)
throw(newArithmeticException());
//方法中抛出异常
c=a/b;
System.out.println("
a/b="
+c);
}
catch(ArrayIndexOutOfBoundsExceptione)
{System.out.println("
没有命令行第一个参数"
catch(ArithmeticExceptione)
算数运算错误"
9.4自定义异常类
虽然Java已经预定义了很多异常类,但有的情况下,程序员不仅需要自己抛出异常,还要创建自己的异常类。
这时可以通过创建Exception的子类来定义自己的异常类。
下面给出一些原则,提示读者何时需要自定义异常类。
满足下列任何一种或多种情况就应该考虑自己定义异常类。
1.Java异常类体系中不包含所需要的异常类型。
2.用户需要将自己所提供类的异常与其他人提供的异常进行区分。
3.类中将多次抛出这种类型的异常。
4.如果使用其它程序包中定义的异常类,将影响程序包的独立性与自包含性。
【例9-6】自定义异常类。
此例中定义了一个异常类MyException,该类是java.lang.Exception类的子类,只包含了两个简单的构造方法。
UsingMyexception类包含了两个方法f()和g(),这两个方法中分别声明并抛出了MyException类型的异常。
在TestMyExceptionlei类的main()方法中,访问了UsingMyException类的f()和g(),并用try-catch语句实现了异常处理。
在捕获了f()和g()抛出的异常后,将在相应的Catch语句块中输出异常的信息,并输出异常发生位置的堆栈跟踪轨迹。
classMyExceptionextendsException//自定义异常类
MyException(){}
MyException(Stringmsg)
{super(msg);
classUsingMyException//抛出异常类
Voidf()throwsMyException
ThrowsMyExceptionfromf()"
thrownewMyException();
Voidg()throwsMyException
ThrowsMyExceptionfromg()"
thrownewMyException("
originateding()"
publicclassTestMyException//捕获并处理异常
UsingMyExceptionm=newUsingMyException();
//创建自定义异常类对象
m.f();
catch(MyExceptione){
m.g();
图9-6
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Java 异常 处理 基础知识
![提示](https://static.bdocx.com/images/bang_tan.gif)