异常
如果由于程序的错误或一些外部环境的影响,导致用户在运行程序期间所做的工作丢失,程序应该做到以下几点:
- 向用户通知错误;
- 返回到一种安全状态,并能让用户执行其他命令;或者
- 允许用户保存所有工作的结果并以妥善的方式退出程序。
异常
- 异常有两种类型:检查型和非检查型异常。
处理错误
为了能够处理程序中的异常情况,需要考虑程序中可能出现的错误和问题,通常包括:
- 用户输入错误;
- 设备错误(如打印机可能在打印过程中没纸了或则卡纸了);
- 物理限制(如磁盘已满);
- 代码错误。
异常分类
- 所有的异常都是由Throwable继承而来
-
Error类:Java运行时系统的内部错误和资源耗尽错误。【这种情况很少出现】
-
Exception类:需要重点关注;
- RuntimeException:由编译错误导致的异常:
- 错误的强制类型转换;
- 数组越界访问;
- 访问null指针
- IOException:程序本身没有问题,但由于I/O错误导致的异常:
- 试图打开一个不存在的文件;
- 试图超越文件末尾继续读取数据;
- 试图根据给定的字符串查找Class对象,而这个字符串表示的类并不存在。
- RuntimeException:由编译错误导致的异常:
-
“如果出现RuntimeException异常,那肯定是你的问题”规则。
-
所有派生于Error类或RuntimeException类的异常都属于非检查型异常,其他则为检查型异常。编译器将检查你是否为所有的检查型异常提供了异常处理器。
-
Java的非检查型异常类:
异常 | 描述 |
---|---|
ArithmeticException | 当出现异常的运算条件时,抛出此异常。例如,一个整数"除以零"时,抛出此类的一个实例。 |
ArrayIndexOutOfBoundsException | 用非法索引访问数组时抛出的异常。如果索引为负或大于等于数组大小,则该索引为非法索引。 |
ArrayStoreException | 试图将错误类型的对象存储到一个对象数组时抛出的异常。 |
ClassCastException | 当试图将对象强制转换为不是实例的子类时,抛出该异常。 |
IllegalArgumentException | 抛出的异常表明向方法传递了一个不合法或不正确的参数。 |
IllegalMonitorStateException | 抛出的异常表明某一线程已经试图等待对象的监视器,或者试图通知其他正在等待对象的监视器而本身没有指定监视器的线程。 |
IllegalStateException | 在非法或不适当的时间调用方法时产生的信号。换句话说,即 Java 环境或 Java 应用程序没有处于请求操作所要求的适当状态下。 |
IllegalThreadStateException | 线程没有处于请求操作所要求的适当状态时抛出的异常。 |
IndexOutOfBoundsException | 指示某排序索引(例如对数组、字符串或向量的排序)超出范围时抛出。 |
NegativeArraySizeException | 如果应用程序试图创建大小为负的数组,则抛出该异常。 |
NullPointerException | 当应用程序试图在需要对象的地方使用 null 时,抛出该异常 |
NumberFormatException | 当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常。 |
SecurityException | 由安全管理器抛出的异常,指示存在安全侵犯。 |
StringIndexOutOfBoundsException | 此异常由 String 方法抛出,指示索引或者为负,或者超出字符串的大小。 |
UnsupportedOperationException | 当不支持请求的操作时,抛出该异常。 |
- Java 定义在 java.lang 包中的检查型异常类:
异常 | 描述 |
---|---|
ClassNotFoundException | 应用程序试图加载类时,找不到相应的类,抛出该异常。 |
CloneNotSupportedException | 当调用 Object 类中的 clone 方法克隆对象,但该对象的类无法实现 Cloneable 接口时,抛出该异常。 |
IllegalAccessException | 拒绝访问一个类的时候,抛出该异常。 |
InstantiationException | 当试图使用 Class 类中的 newInstance 方法创建一个类的实例,而指定的类对象因为是一个接口或是一个抽象类而无法实例化时,抛出该异常。 |
InterruptedException | 一个线程被另一个线程中断,抛出该异常。 |
NoSuchFieldException | 请求的变量不存在 |
NoSuchMethodException | 请求的方法不存在 |
声明检查型异常
-
一个方法必须声明所有可能抛出的检查型异常;
-
声明的时候尽可能声明具体的异常,方便更好的处理;
-
当前方法不知道如何处理这种异常,可将该异常交给上一级调用者来处理(非RuntimeException类型的异常);
-
如果在子类中覆盖了超类的一个方法,子类方法中声明的检查型异常不能比超类声明的异常更通用(子类方法可以抛出更特定的异常或者根本不抛异常)。如果超类方法没有抛任何检查型异常,则子类也不能抛出任何检查型异常。
格式:
public 返回值类型 方法名(参数列表...)
throws 异常类A,异常类B... {
}
如何抛出异常
-
抛出异常的语句:
throw new [异常类]([参数]);
或者:
var e = new [异常类]([参数]); throw e;
-
可以抛出的异常必须是Throwable或其子类的实例。
-
throw 和throws关键字的区别:
throw用于抛出异常对象,后面跟的是异常对象;throw用在方法内。
throws用于抛出异常类,后面跟的异常类名,可以跟多个,用逗号隔开;throws用在方法上。
创建自己的异常类
当遇到任何标准异常类都无法描述的问题时,可以创建自己的异常类,即定义一个派生于Exception的类或派生于Exception某个子类的类。自定义的异常类应当包含两个构造器:一个默认构造器和一个包含详细描述信息的构造器(超类Throwable的ToString方法会返回一个字符串,其中包含这个详细描述信息,这在调试时非常有用)。
class FileFormatException() extends IOException{
public FileFormatException() {} //默认构造器
public FileFormatException(String gripe) { //包含详细描述信息的构造器
super(gripe);
}
}
捕获异常
格式:
try{
可能会发生的异常
}catch(异常类型 异常名(变量)){
针对异常进行处理的代码
}catch(异常类型 异常名(变量)){
针对异常进行处理的代码
}...
[finally{
释放资源代码;
}]
注意:
-
catch 不能独立于 try 存在。
-
catch里面不能没有内容
-
在 try/catch 后面添加 finally 块并非强制性要求的。
-
try 代码后不能既没 catch 块也没 finally 块。
-
try里面越少越好。
-
try, catch, finally 块之间不能添加任何代码。
-
finally里面的代码最终一定会执行(除了JVM退出)
-
如果程序可能存在多个异常,需要多个catch进行捕获。
-
异常如果是同级关系,catch谁前谁后没有关系
如果异常之间存在上下级关系,上级需要放在后面 -
异常对象可能包含有关异常性质的信息,可以使用
e.message()
获取或者使用e.getClass.getName()
获取异常对象的实际类型。