目录
- 异常的概念
- 异常的体系结构
- 异常
- 抛出异常
- 处理异常
- throws声明
- try-catch捕获
- 异常处理流程
- 自定义异常类
异常的概念
Java中,程序执行过程中发生的不正常的行为称为异常。
我们之前学数组的时候可能会遇到的数组越界异常:ArrayIndexOutOfBoundsException
学对象和引用类型时会遇到空指针异常:java.lang.NullPointerException
学运算符时可能会遇到算术异常: java.lang.ArithmeticException
异常的体系结构
- 最上面的Throwable为顶层类,派生出两个子类:Error和Exception
- Errror:这个指错误,Java虚拟机也无法解决,一般有StackOverflowError和OutOfMemoryError。
- Exception,程序员可以对代码进行处理,使程序继续执行。
异常
异常分为两类:编译时异常(受查异常)和运行时异常(非受查异常)
- RunTimeException以及其子类对应的异常,都称为运行时异常。比如:NullPointerException、ArrayIndexOutOfBoundsException、ArithmeticException。
- 编译时语法错误不可以称为异常
- 异常处理主要的5个关键字:throw、try、catch、final、throws。
抛出异常
可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。语法如下:
throw new XXXException("异常产生的原因");
- throw必须写在方法内部
- 抛出的必须为Exception或者Exception的子类
- 如果抛出的为运行时异常,程序员不用进行处理,交给jvm处理了
- 如果抛出的是编译时异常,用户必须进行处理,否则无法编译
- 异常一旦抛出,其后面代码无法执行
处理异常
异常的处理有两种方法:一种是使用throws声明一下,一种是使用try-catch进行捕获处理
throws声明
处在方法声明时参数列表之后,当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常。
语法如下:
修饰符 返回值类型 方法名(参数列表) throws 异常类型1,异常类型2…{
}
- throws必须跟在方法列表后面
- 声明的异常必须是Exception或者Exception子类
- 如果方法中抛出了多个异常,我们就要声明多个异常,如果异常之间具有父子关系则只用声明父类就行
- 调用声明抛出函数时,调用的函数必须处理异常或者继续声明
try-catch捕获
throws对异常并没有真正处理,而是将异常报告给抛出异常方法的调用者,由调用者处理。如果真正要对异常进行处理,就需要try-catch。
语法格式:
try{
// 将可能出现异常的代码放在这里
}catch(要捕获的异常类型 e){
// 如果try中的代码抛出异常了,此处catch捕获时异常类型与try中抛出的异常类型一致时,或者是try中抛出异常的基类时,就会被捕获到
// 对异常就可以正常处理,处理完成后,跳出try-catch结构,继续执行后序代码
}[catch(异常类型 e){
// 对异常进行处理
}finally{
// 此处代码一定会被执行到
}]
// 后序代码
// 当异常被捕获到时,异常就被处理了,这里的后序代码一定会执行
// 如果捕获了,由于捕获时类型不对,那就没有捕获到,这里的代码就不会被执行
注意:
- []中表示可选项,可以添加,也可以不用添加
- try中的代码可能会抛出异常,也可能不会
- try块内抛出异常位置之后的代码将不会被执行
- 如果抛出异常类型与catch时异常类型不匹配,即异常不会被成功捕获,也就不会被处理,继续往外抛,直到JVM收到后中断程序----异常是按照类型来捕获的
- try中可能会抛出多个不同的异常对象,则必须用多个catch来捕获----即多种异常,多次捕获
- 如果多个异常处理方式相同,我们也可以这样写:
catch (ArrayIndexOutOfBoundsException | NullPointerException e) {
...
}
- 如果异常之间具有父子关系,一定是子类异常在前catch,父类异常在后catch,否则语法错误
- finally中的代码一定会执行的,一般在finally中进行一些资源清理的扫尾工作。
异常处理流程
- 程序先执行 try 中的代码
- 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
- 如果找到匹配的异常类型, 就会执行 catch 中的代码
- 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
- 如果上层调用者也没有处理的了异常, 就继续向上传递.
- 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.
自定义异常类
对于Java给我们提供的异常有时可能并不能满足某些场景,这就需要我们自定义异常类。定义方式如下:
- 自定义异常类,然后继承自Exception 或者 RunTimeException
- 实现一个带有String类型参数的构造方法,参数含义:出现异常的原因
举个例子:
class UserNameException extends Exception {
public UserNameException(String message) {
super(message);
}
}
if (!userName.equals(userName)) {
throw new UserNameException("用户名错误!");
}