目录
异常的处理方式之一:捕获异常
try-catch-finally 语句块的执行过程:
异常的处理方式之二:声明异常(throws 子句)
自定义异常
异常的处理方式之一:捕获异常
捕获异常是通过 3 个关键词来实现的:try-catch-finally。用 try 来执行一段程序,如果 出现异常,系统抛出一个异常,可以通过它的类型来捕捉(catch)并处理它,最后一步是 通过 finally 语句为异常处理ᨀ供一个统一的出口,finally 所指定的代码都要被执行(catch 语句可有多条;finally 语句最多只能有一条,根据自己的需要可有可无)。
上面过程详细解析:
- try:
try 语句指定了一段代码,该段代码就是异常捕获并处理的范围。在执行过程中,当任 意一条语句产生异常时,就会跳过该条语句中后面的代码。代码中可能会产生并抛出一种或 几种类型的异常对象,它后面的 catch 语句要分别对这些异常做相应的处理。
一个 try 语句必须带有至少一个 catch 语句块或一个 finally 语句块 。
当异常处理的代码执行结束以后,不会回到 try 语句去执行尚未执行的代码。 - catch:
每个 try 语句块可以伴随一个或多个 catch 语句,用于处理可能产生的不同类 型的异常对象。
常用方法,这些方法均继承自 Throwable 类 。
toString ()方法,显示异常的类名和产生异常的原因
getMessage()方法,只显示产生异常的原因,但不显示类名
printStackTrace()方法,用来跟踪异常事件发生时堆栈的内容
catch 捕获异常时的捕获顺序:
如果异常类之间有继承关系,在顺序安排上需注意。越是顶层的类,越放 在下面,再不然就直接把多余的 catch 省略掉。 也就是先捕获子类异常 再捕获父类异常。 - finally:
有些语句,不管是否发生了异常,都必须要执行,那么就可以把这样的语句放 到 finally 语句块中。
通常在 finally 中关闭程序块已打开的资源,比如:关闭文件流、释放数据库 连接等。
try-catch-finally 语句块的执行过程:
程序首先执行可能发生异常的 try 语句块。如果 try 语句没有出现异常则执行完后跳至 finally 语句块执行;如果 try 语句出现异常,则中断执行并根据发生的异常类型跳至相应的 catch 语句块执行处理。catch 语句块可以有多个,分别捕获不同类型的异常。catch 语句块 执行完后程序会继续执行 finally 语句块。finally 语句是可选的,如果有的话,则不管是否发 生异常,finally 语句都会被执行。
即使 try 和 catch 块中存在 return 语句,finally 语句也会执行。是在执行完 finally 语句 后再通过 return 退出。
finally 语 句 块 只 有 一 种 情 况 是 不 会 执 行 的 , 那 就 是 在 执 行 finally 之 前 遇 到 了 System.exit(0)结束程序运行。
异常处理的典型代码(捕获异常)
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
public class Test {
public static void main(String[ ] args) {
FileReader reader = null;
try {
reader = new FileReader("d:/a.txt");
char c = (char) reader.read();
char c2 = (char) reader.read();
System.out.println("" + c + c2);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (reader != null) {
reader.close();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
异常的处理方式之二:声明异常(throws 子句)
当 CheckedException 产生时,不一定立刻处理它,可以再把异常 throws 出去
在方法中使用 try-catch-finally 是由这个方法来处理异常。但是在一些情况下,当前方法 并不需要处理发生的异常,而是向上传递给调用它的方法处理。
如果一个方法中可能产生某种异常,但是并不能确定如何处理这种异常,则应根据异常 规范在方法的首部声明该方法可能抛出的异常。
如果一个方法抛出多个已检查异常,就必须在方法的首部列出所有的异常,之间以逗号 隔开。
异常处理的典型代码(声明异常抛出 throws)
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
public class Test {
public static void main(String[ ] args) {
try {
readFile("joke.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();
}
}
}
方法重写中声明异常原则:子类重写父类方法时,如果父类方法有声明异常,那么子类 声明的异常范围不能超过父类声明的范围。
自定义异常
- 在程序中,可能会遇到 JDK ᨀ供的任何标准异常类都无法充分᧿述清楚我们想要 表达的问题,这种情况下可以创建自己的异常类,即自定义异常类。
- 自定义异常类只需从 Exception 类或者它的子类派生一个子类即可。
- 自定义异常类如果继承 Exception 类,则为受检查异常,必须对其进行处理;如果 不想处理,可以让自定义异常类继承运行时异常 RuntimeException 类。
- 习惯上,自定义异常类应该包含 2 个构造器:一个是默认的构造器,另一个是带有 详细信息的构造器。
自定义异常类
/**IllegalAgeException:非法年龄异常,继承 Exception 类*/
class IllegalAgeException extends Exception {
//默认构造器
public IllegalAgeException() {
}
//带有详细信息的构造器,信息存储在 message 中
public IllegalAgeException(String message) {
super(message);
}
}
自定义异常类的使用
class Person {
private String name;
private int age;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) throws IllegalAgeException {
if (age < 0) {
throw new IllegalAgeException("人的年龄不应该为负数");
}
this.age = age;
}
public String toString() {
return "name is " + name + " and age is " + age;
}
}
public class TestMyException {
public static void main(String[ ] args) {
Person p = new Person();
try {
p.setName("Lincoln");
p.setAge(-1);
} catch (IllegalAgeException e) {
e.printStackTrace();
System.exit(-1);
}
System.out.println(p);
}
}
使用异常机制的建议
要避免使用异常处理代替错误处理,这样会降低程序的清晰性,并且效率低下处理异常不可以代替简单测试---只在异常情况下使用异常机制
不要进行小粒度的异常处理---应该将整个任务包装在一个 try 语句块中
异常往往在高层处理