目录
异常的产生
- 返回到一种安全状态,并能够让用户执行一些其他的命令
-
允许用户保存所有操作的结果 , 并以妥善的方式终止程序
- 用户输入的错误
- 设备错误
- 物理限制
- 代码错误
异常的分类
java中的异常的类都来源于java.lang的包中,这是Java程序会自动导入的包,不需要手动import
Java中的异常都派生于Throwable类,可以分为两大类:
- 运行时异常,也成为非检查型异常
- 非运行时异常,也成为检查时异常
Throwable类
Throwable类是所有异常类的子类,所有的异常对象都是派生于Throwable类的实例。Throwable类在异常界的地位就像Object类在所有类中的地位。
Error类(非检查型异常)
Error在英语中是错误的意思,Error类描述了运行时内部错误和资源耗尽的错误。
程序不应该抛出这种类型的对象,这种错误一旦出现,一般的解决办法是妥善的终止程序,程序几乎不能处理,属于大问题。
常见的Error异常 Error错误的全部类
IOError | 当出现严重的I/O异常时抛出 |
ThreadError | 在受害的线程中抛出 |
RuntimeException类(非检查型异常)
由编译错误导致的异常属于RuntimeException异常。常常包括以下问题:
- 错误的强制类型转换
- 数组的越界访问
- 指针指向空指针,访问空指针
常见的RuntimeException异常 RuntimeException异常的全部类
ArithmeticException | 数字计算的异常。例如除数为0会抛出此异常 |
NullPointerException | 空指针异常 |
ArrayOutOfBoundsException | 数组越界产生的异常 |
ClassNotFoundException | 找不到类文件的异常 |
ClassCastException | 类文件强制转化的异常 |
数字计算错误的异常举例:
try{
int i = 3 ;
int j = 0 ;
System.out.println(i/j);
catch(ArithmeticException e){
System.out.println("除数不能为0!");
}
}
IOException类(检查型异常)
输入输出等找不到路径的异常属于IOException异常。常常包括以下问题:
- 试图超越文件末尾继续读取文件
- 试图打开一个不存在的文件
- 属兔查找一个不存在的类
常见的IOException类异常 IOException类的全部子类
FileNotFoundException | 文件不存在时抛出的异常 |
IIOException | 读写操做的异常 |
EOFException | 遇到文件或流的结尾的异常 |
文件的异常举例:
try{
var in =new FileInputStream(filename);
int b;
while((b=in.read())!=-1)
{
//代码段
}
}
carch(IOException e){
//打印一个堆栈的轨迹
e.printStackTrace();
}
}
异常的处理方法
一、try--catch--finally方法
try{ //代码段 }
catch(异常的类名 对象名){ //代码处理段 }
........
catch(异常的类名 对象名){ //代码处理段 }
finally{ // 代码段}
- 其中try和catch必须同时出现 ,但是finally可以不出现
- try中的语句是必须运行的,在try中可能会有异常的抛出
- catch中接受try中抛出的异常,可能没有异常,则不执行catch的代码
- finally是必须执行的代码段,不论有没有异常
- catch可以有多个,用来处理不同类型的异常
从中可以看到try-catch-finally语句是可以嵌套的,即可以有异常中的异常出现
try{
try{ //代码段 }
catch(异常的类名 对象名){ //代码处理段 }
finally{ // 代码段}
}
catch(异常的类名 对象名){ //代码处理段 }
........
catch(异常的类名 对象名){ //代码处理段 }
finally{
//代码段
try{ //代码段 }
catch(异常的类名 对象名){ //代码处理段 }
finally{ // 代码段}
}
throw抛出异常
如果需要在程序中自行抛出异常,则应使用 throw 语句,throw 语句抛出是一个异常实例,而且每次只能抛出一个异常实例。
try{ //代码段}
catch(Exception e){ throw e...} ......
catch(SQLException e){
throw new ServleException("error"+e.getMassage());
}
//可以改写为
catch(SQLException e){
var es=new ServleException("error");
es.initCause(e);
throw es;
}
二、用throws声明,将异常传递给调用者
不用写出具体的异常处理方法,而是将其传播,但必须在方法的首部添加一个 throws 说明符,提醒调用者这个方法可能抛出的异常
上面文件的IOException的异常就可以改写为:
public void read(String filename) throws IOException{
var in =new FileInputStream(filename);
int b;
while((b=in.read())!=-1)
{
//代码段
}
虽然这个方法不叫简洁,但是有几个要注意的点:
- 如果编写一个方法覆盖超类方法,而这个超类方法没有抛出异常,就必须捕获你的方法中出现的每一个检查型异常
- 不允许在子类的throws中出现超类中方法没有出现的异常
throws和throw的区别
- throws用于异常方法的声明,throw在方法体内使用
- throws后面是异常类型,可以有多个;throw后面是异常的实例对象,只能有一个
- throws用来声明,由调用者处理,不一定会出现异常;throw是出现异常时手动来结束方法的,执行了throw一定会出现异常
使用异常的技巧
-
异常的处理不能代替简单的测试
例如我们要强制要求空栈弹出若干次,首先判断是否为空栈
if(!s.empty())
s.pop();
而用异常的方法则要求我们不能这样做
try{
s.pop();
}
catch(EmptyStackException e){}
isempty方法的运行时间远小于异常的方法,因此,异常仅在异常的情况下使用
2.不要尽量细化异常
细化异常会使代码的量急剧增加,运行效率会降低
//例如将try嵌套在for循环中
for(int i = 0; i < 100 ; i ++){
try{ s.pop()}
catch(....){}
//尽量写成
try{
for(int i = 0 ; i < 100 ; i ++){
s.pop();
}
catch(......){}
将整个任务包在一个try中,这样出现问题就可以取消整个程序
3.充分利用异常的层次结构
- 不要只抛出RuntimeException异常,找寻一个适合的子类或子集创建一个子类
- 不要只捕获Throwable异常,否则代码很难维护
4.宁愿严格处理异常,也不要放任