异常机制
1、概念
异常指程序运行过程中出现的非正常现象,例如用户输入错误、除数为零、需要处理的文件不存在、数组下标越界等。在Java的异常处理机制中,引进了很多用来描述和处理异常的类,称为异常类。
2、异常的分类
Throwable下面又派生了两个子类:Error和Exception。
2.1、Error(错误)
Error是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。
2.2、Exception(异常)
Exception是程序本身能够处理的异常,如:空指针异常(NullPointerException)、数组下标越界异常(ArrayIndexOutOfBoundsException)、类型转换异(ClassCastException)、算术异常(ArithmeticException)等。
- 运行时异常:RuntimeException
- 已检测异常:CheckedException
派生于RuntimeException的异常,如被 0 除、数组下标越界、空指针等,其产生比较频繁,处理麻烦,如果显式的声明或捕获将会对程序可读性和运行效率影响很大。
- ArithmeticException异常:试图除以0
- NullPointerException异常:空指针异常
- ClassCastException异常:类型转换异常
- ArrayIndexOutOfBoundsException异常:数组下标越界
- NumberFormatException异常:出现数字格式异常
所有不是RuntimeException的异常,统称为Checked Exception,又被称为“已检查异常”,如IOException、SQLException等以及用户自定义的Exception异常。 这类异常在编译时就必须做出处理,否则无法通过编译。
3、异常处理方法之一:捕获异常
捕获异常是通过3个关键词来实现的:try-catch-finally。
1、try:try语句指定了一段代码,该段代码就是异常捕获并处理的范围。
一个try语句必须带有至少一个catch语句块或一个finally语句块 。
注意:当异常处理的代码执行结束以后,不会回到try语句去执行尚未执行的代码。
- toString ()方法,显示异常的类名和产生异常的原因。
- getMessage()方法,只显示产生异常的原因,但不显示类名。
- printStackTrace()方法,用来跟踪异常事件发生时堆栈的内容。
catch的捕获顺序:如果异常类之间有继承关系,在顺序安排上需注意。越是顶层的类,越放在下面,再不然就直接把多余的catch省略掉。 也就是先捕获子类异常再捕获父类异常。
不管是否发生了异常,都必须要执行,那么就可以把这样的语句放到finally语句块中。
注意:finally语句块只有一种情况是不会执行的,那就是在执行finally之前遇到了System.exit(0)结束程序运行。
4、异常的处理方式之二:声明异常(throws子句)
当CheckedException产生时,不一定立刻处理它,可以再把异常throws出去。
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;public class Test9 {public static void main(String[] args) {try {readFile("test.txt");} catch (FileNotFoundException e) {System.out.println("所需文件不存在!");} catch (IOException e) {System.out.println("文件读写错误!");}} public static void readFile(String fileName) throws FileNotFoundException,IOException {FileReader in = new FileReader(fileName);int tem = 0;try {tem = in.read();while (tem != -1) {System.out.print((char) tem);tem = in.read();}} finally {in.close();}}
}
5、自定义异常
在程序中,可能会遇到JDK提供的任何标准异常类都无法充分描述清楚我们想要表达的问题,这种情况下可以创建自己的异常类,即自定义异常类。
1、自定义异常类只需从Exception类或者它的子类派生一个子类即可。(也就是说只需要继承Exception即可)。
2.自定义异常类如果继承Exception类,则为受检查异常,必须对其进行处理;如果不想处理,可以让自定义异常类继承运行时异常RuntimeException类。
3.习惯上,自定义异常类应该包含2个构造器:一个是默认的构造器,另一个是带有详细信息的构造器。
package com.dx1;
public class TestExcption {public static void main(String[] args) {int i1=1000;int i2=1000;if(i1==i2) {throw new TestException("测试异常!!");}
}
}
//测试异常
class TestException extends RuntimeException{/*** */private static final long serialVersionUID = 1L;public TestException() {}public TestException(String message) {super(message);}
}
总结: 总的来说,处理异常的方式有三种分别为:捕获异常try{}catch{}finally{}、声明异常 throws 、自定义异常。