异常介绍
java 语言中,将程序执行过程中发生的不正常情况称为“异常”,开发过程中的语法错误和逻辑错误不是异常。
执行过程中所发生的的异常事件可分为两大类:
- Error (错误):java 虚拟机无法解决的严重问题,如 JVM 系统内部错误、资源耗尽等严重情况。比如:StackOverflowError(栈溢出)和OOM(out of memory,内存不足)。Error 是严重错误,程序会崩溃
- Exception :其他因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理,例如空指针访问,视图读取不存在的文件、网络连接中断等等。 Exception 分为两大类:运行时异常(程序运行时,发生的异常,即在运行 .class 字节码文件时出现的异常,发生在运行阶段)和编译时异常(编程时,编译器检查出来的异常,即在进行 javac 编译时发现的异常,发生在编译阶段)
异常体系结构图
- 异常分为两大类,运行时异常和编译时异常
- 运行时异常,编译器检查不出来,一般是指编程时的逻辑错误,是程序员应该避免其出现的异常, java.lang.RuntimeException 类及它的子类都是运行时异常
- 对于运行时异常,可以不作处理,因为这类异常很普遍,若全处理可能会对程序的可读性和运行效率产生影响
- 编译时异常必须进行处理,否则不能通过编译,也不能运行
常见的运行时异常
- NullPointerException (空指针异常):当应用程序试图在需要对象的地方使用 null 时,抛出该异常
- ArithmeticException (数学运算异常):当出现异常的运算条件时,抛出此异常。例如,一个整数除以零时抛出此异常
- ArraysIndexOutOfBoundException (数组越界异常):用非法索引访问数组时抛出的异常。如果索引为负或大于等于数组大小,则该索引为非法索引
- ClassCastException (类型转换异常):当试图将对象强制转换为不是实例的子类时,抛出该异常
- NnumberFormatException (数字格式不正确异常):当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常(可以使用该异常保证输入的是满足条件的数字)
常见的编译异常
编译异常是指在编译期间,就必须处理的异常,否则代码不能通过编译
- SQLException :操作数据库时,查询表可能发生异常
- IOException :操作文件时发生的异常
- FileNotFoundException :操作一个不存在的文件时发生的异常
- ClassNotFoundException :加载类,而该类不存在时发生的异常
- EOFException :操作文件到文件末尾发生的异常
- IllegalArguementException :参数异常
异常处理机制
异常处理就是当异常发生时,对异常处理的方式。如果程序员没有显式的处理异常,默认就是 throws
try-catch-finally
程序员在代码中捕获发生的异常,自行处理。try 块用于包含可能发生异常的代码, catch 块用于处理 try 块中发生的异常,可以根据需要在程序中有多个 try-catch 块
基本语法
try {
// 可能出现异常的代码块
// 将异常生成对象的异常对象,传递给 catch 块
} catch (异常){
// 对异常的处理
}
// 可以没有 finally 块
示意图
注意细节:
- 如果异常发生了,则异常所在位置的后续代码不会执行,直接进入到 catch 块
- 如果异常没有发生,则顺序执行 try 块,不会进入 catch 块
- 如果希望不管是否发生异常,都执行某段代码(如关闭连接、释放资源等),则使用 finally 块
- 可以有多个 catch 语句,捕获不同的异常(进行不同的业务处理),要求父类异常在后,子类异常在前,比如 Exception 在后, NullPointerException 在前。如果发生异常,只会匹配到一个 catch
try {
} catch (NullPointerException e ) {
} catch (Exception e) {
} finally {
}
- 可以进行 try-catch 配合使用,这用法相当于没有捕获异常,因此程序会直接崩掉退出。这种用法的应用场景就是执行一段代码时,不管这段代码是否发生异常,都必须执行某个业务逻辑
try {
// 执行某段代码
} finally {
// 必须执行的某个业务逻辑
}
try-catch 的实际应用
让用户输入一个整数,如果用户输入的不是整数,就进行提示,直到他输入正确
import java.util.Scanner;
public class Test {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
System.out.println("请输入一个整数:");
int num = 0;
while(true) {
try {
num = Integer.parseInt(scanner.next());
break;
} catch (Exception e) {
System.out.println("输入的不是整数,请重新输入:");
}
}
System.out.println("您输入的整数是:" + num);
}
}
throws
将发生的异常抛出,交给调用者来处理,最顶级的处理者是 JVM。
如果一个方法可能生成某种异常,但是并不能确定如何处理这种异常(即方法内没有进行 try-catch 处理),则此方法应显示的声明抛出异常,表明该方法将不对这些异常进行处理,而由方法的调用者负责处理
在方法声明中用 throws 语句可以声明抛出异常的列表。throws 后面的异常类型可以是方法中产生的异常类型,也可以是它的父类。
// throws 后面可以是方法中产生的异常类型 FileNotFoundException
// 也可以是它的父类 Exception
// 还可以是异常列表 FileNotFoundException,NullPointerException,ArithmeticException
public void f2() throws FileNotFoundException {
FileInputStream file = new FileInputStream("文件地址");
}
注意事项
- 对于编译异常,程序中必须处理,比如 try-catch 或 throws ,没有默认处理
- 对于运行时异常,程序中如果没有处理,默认就是 throws的方式处理
- 子类重写父类的方法时,对抛出异常的规定:子类重写的方法,所抛出的异常类型要么和父类抛出的异常一致,要么为父类抛出异常的子类
class Father {
public void hi() throws Exception {}
}
class Son extends Father {
// 子类重写父类方法后抛出的异常只能 <= 父类方法抛出的有一次
public void hi() throws NullPointerException{}
}
- 在 throws 过程中,如果有方法 try-catch ,就相当于处理异常,就可以不必 throws
自定义异常
基本概念
当程序中出现了某些“错误”,但该错误信息并没有在 Throwable 子类中描述处理,这个时候可以自己设计异常类,用于描述该错误信息。
自定义异常的步骤
- 定义类:自定义异常类名,继承 Exception 或 RuntimeException
- 如果继承 Exception ,属于编译异常
- 如果继承 RuntimeException ,属于运行异常(一般来说会继承 RuntimeException)
public class Test {
public static void main(String[] args) {
int age = 193;
if(!(age > 0 && age <= 120)){
throw new AgeException("年龄只能在-~120岁之间");
}
System.out.println("年龄正确...");
}
}
// 一般情况下,自定义异常继承 RuntimeException ,即把异常定义为运行时异常
// 好处是可以使用默认的 throws 处理机制
// 如果是编译时异常,不能使用默认 throws 处理,需要显式处理
class AgeException extends RuntimeException {
public AgeException(String message){
super(message);
}
}
throw 和 throws 区别
意义 | 位置 | 后面跟的东西 | |
throws | 异常处理的一种方式 | 方法声明处 | 异常类型(异常类) |
throw | 手动生成异常对象的一个关键字 | 方法体中 | 异常对象(new 异常类) |