异常 :指的是程序在执行过程中,出现的非正常的情况,最终会导致JVM的非正常停止。
在Java等面向对象的编程语言中,异常本身是一个类,产生异常就是创建异常对象并抛出了一个异常对象。Java处理异常的方式是中断处理。
异常的根类是 java.lang.Throwable
Throwable体系:
Error:严重错误Error,无法通过处理的错误,只能事先避免。
Exception:表示异常,异常产生后程序员可以通过代码的方式纠正,使程序继续运行,是必须要处理的。
Throwable中的常用方法:
public void printStackTrace() :打印异常的详细信息。JVM打印异常对象,默认此方法
包含了异常的类型,异常的原因,还包括异常出现的位置,在开发和调试阶段,都得使用printStackTrace。
public String getMessage() :获取发生异常的原因。简短描述
提示给用户的时候,就提示错误原因。
public String toString() :获取异常的类型和异常描述信息(不用)。详细消息字符串
异常分类
编译时期异常:checked异常。在编译时期,就会检查,如果没有处理异常,则编译失败。(如日期格式化异常)
运行时期异常:runtime异常。在运行时期,检查异常.在编译时期,运行异常不会编译器检测(不报错)。(如数学异常)
异常的处理
抛出异常throw
在java中,提供了一个throw关键字,它用来抛出一个指定的异常对象。
1. 创建一个异常对象。封装一些提示信息(信息可以自己编写)。
2. 需要将这个异常对象告知给调用者。
Objects非空判断
Objects类由一些静态的实用方法组成,这些方法是null-save(空指针安
全的)或null-tolerant(容忍空指针的)
public static T requireNonNull(T obj) :查看指定引用对象不是null。
查看源码发现这里对为null的进行了抛出异常操作:
public static <T> T requireNonNull(T obj) {
if (obj == null)
throw new NullPointerException();
return obj;
}
声明异常throws
修饰符 返回值类型 方法名(参数) throws 异常类名1,异常类名2…{ }
注意:
1、throws关键字必须写在方法声明处
2、throws关键字后边生命的异常必须是Exception或者是Exception的子类
3、方法内部如果抛出了多个异常对象,那么throws后边必须也声明多个异常
如果抛出的多个异常对象有子父类关系,那么直接声明父类异常即可
4、调用了一个声明抛出异常的方法,我们就必须处理声明的异常
要么继续使用throws声明抛出,交给方法的调用者处理,最终交给JVM
要么try…catch自己处理的异常
捕获异常try…catch
如果异常出现的话,会立刻终止程序,所以我们得处理异常:
1. 该方法不处理,而是声明抛出,由该方法的调用者来处理(throws)。
2. 在方法中使用try-catch的语句块来处理异常。
try{
编写可能会出现异常的代码
}catch(异常类型 e){
处理异常的代码
//记录日志/打印异常信息/继续抛出异常
}
注意:
1、try中可能会抛出多个异常对象,那么就可以使用多个catch来处理这些异常对象
2、如果try中产生了异常,那么就会执行catch中的异常处理逻辑,执行完毕catch中的的处理逻辑,继续执行try…catch之后的代码
如果try中没有产生异常,那么就不会执行catch中异常的处理逻辑,执行完try中的代码,继续执行try…catch之后的代码
finally 代码块
有一些特定的代码无论异常是否发生,都需要执行。另外,因为异常会引发程序跳转,导致有些语句执行不到。而finally就是解决这个问题的,在finally代码块中存放的代码都是一定会被执行的。
fifinally的语法:
try...catch....fifinally:自身需要处理异常,最终还得关闭资源。
的IO流中,当打开了一个关联文件的资源,最后程序不管结果如何,都需要把这个资源关闭掉。
异常注意事项
- 多个异常使用捕获:
1. 多个异常分别处理。
2. 多个异常一次捕获,多次处理。
catch里面定义的异常变量,如果有子父类关系,那么子类的异常变量必须写在上边,否则就会报错。
3. 多个异常一次捕获一次处理。
- 运行时异常被抛出可以不处理。即不捕获也不声明抛出。
- 如果fifinally有return语句,永远返回fifinally中的结果,避免该情况.
- 如果父类抛出了多个异常,子类重写父类方法时,抛出和父类相同的异常或者是父类异常的子类或者不抛出异常。
- 父类方法没有抛出异常,子类重写父类该方法时也不可抛出异常。此时子类产生该异常,只能捕获处理,不能声明抛出
异常类如何定义:
1. 自定义一个编译期异常: 自定义类 并继承于 java.lang.Exception 。那么自定义的异常类就是一个编译器异常,如果方法内部抛出了编译器异常,就必须处理这个异常,要么throws,要么try…catch
2. 自定义一个运行时期的异常类:自定义类 并继承于 java.lang.RuntimeException 。