作者:困了电视剧
专栏:《JavaSE语法与底层详解》
文章分布:这是一篇关于Java中异常类的文章,在本篇文章中详细讲解了异常的使用逻辑和底层的执行过程,如有疏漏,欢迎大佬指正!
目录
异常的体系结构
异常的分类
编译时异常
运行时异常
异常的两种处理方式
异常发生前避免异常发生
异常发生后处理异常
异常的抛出和捕获
throw关键字
异常的声明throws
try-catch-finally关键字
异常的体系结构
异常的本质是类,Java中不同类型的异常都有,都有与其对应的类来进行描述。由于在编程过程中我们会遇到各种各样的异常,所以为了对不同异常或者错误进行很好的分类管理,Java内部维护了一个异常的体系结构。
1. Throwable : 是异常体系的顶层类,其派生出两个重要的子类 , Error 和 Exception2. Error : 指的是 Java 虚拟机无法解决的严重问题,比如: JVM 的内部错误、资源耗尽等 ,典型代表: StackOverflflowError 和 OutOfMemoryError ,一旦发生回力乏术。3. Exception : 异常产生后程序员可以通过代码进行处理,使程序继续执行。我们平时所说的异常就是 Exception 。
异常的分类
编译时异常
在程序编译期间发生的异常,称为编译时异常,也称为受检查异常(Checked Exception)。举个栗子:
我想使用clone方法,在我实现了Cloneable接口并重写clone方法时,如果我没有对异常进行声明,则会在我编译时就提醒我有异常。
运行时异常
在程序执行期间发生的异常,称为运行时异常,也称为非受检查异常(Unchecked Exception)
RunTimeException以及其子类对应的异常,都称为运行时异常。比如:NullPointerException、
ArrayIndexOutOfBoundsException、ArithmeticException。
注意:编译时出现的语法性错误,不能称之为异常。例如将 System.out.println 拼写错了 , 写成了 system.out.println. 此时编译过程中就会出错 , 这是 " 编译期 " 出错。而运行时指的是程序已经编译通过得到 class 文件了 , 再由 JVM 执行过程中出现的错误。
像数组越界这种,编译时没有报错运行时才会出错的异常就是运行时异常。
异常的两种处理方式
既然不可避免地会出现异常这种问题,那我们就应该思考怎样去解决它,现在一共提供了两种处理方式:
异常发生前避免异常发生
class Test {
void test(){}
void action(){
boolean flag=isOK();
if ( flag==true ){
test();
}else {
return;
}
}
boolean isOK(){
boolean flag=false;
return flag;
}
}
这种方法就是在执行之前先判断这个有没有异常,如果没有异常在进行执行。
异常发生后处理异常
class Test {
void test(){}
void action(){
try{
test();
}catch(Exception e){
e.printStackTrace();
}
}
}
这种方法是我先执行,如果有异常我再进行解决。
在异常发生前进行预防,不仅会造成代码的冗长,并且判断语句和功能语句混杂在一起,当未来想对代码进行维护时会有非常大的困难,所以我们在对异常进行处理时一般采用第二种方法。
异常的抛出和捕获
throw关键字
在编写程序时,如果程序中出现错误,此时就需要将错误的信息告知给调用者,比如:参数检测。
在Java中,可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。
此时我们假定出错条件是当传进来的数不等于0时,在这时我们需要抛出一个异常,将错误信息调用出来并且解决。
异常的声明throws
我们在抛出异常的时候难免会需要抛出编译型异常,这时候java出于安全性等因素的考虑会使程序无法通过编译,此时我们就需要用throws进行异常的声明。
注意:
1. throws必须跟在方法的参数列表之后
2. 声明的异常必须是 Exception 或者 Exception 的子类
3. 方法内部如果抛出了多个异常,throws之后必须跟多个异常类型,之间用逗号隔开,如果抛出多个异常类型具有父子关系,直接声明父类即可4. 调用声明抛出异常的方法时,调用者必须对该异常进行处理,或者继续使用throws抛出
可以通过编译。
try-catch-finally关键字
throws对异常并没有真正处理,而是将异常报告给抛出异常方法的调用者,由调用者处理。如果真正要对异常进行处理,就需要try-catch。
class Test{
public void test01(){
throw new ArithmeticException("01");
}
public void test02(){
throw new ClassCastException("02");
}
public void test03(){
throw new IllegalArgumentException("03");
}
public void test04(){
throw new IllegalStateException("04");
}
public void action(){
try {
test01();
test02();
test03();
test04();
}catch (ArithmeticException e){
e.printStackTrace();
}catch (ClassCastException e){
e.printStackTrace();
}catch (IllegalArgumentException e){
e.printStackTrace();
}catch (IllegalStateException e){
e.printStackTrace();
}finally {
System.out.println("finally中的内容");
}
}
}
结果为:
所以我们可以理解为:
程序先执行 try 中的代码
如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
如果找到匹配的异常类型, 就会执行 catch 中的代码
如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
如果上层调用者也没有处理的了异常, 就继续向上传递.
一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止
以上就是本篇博客的全部内容,如果对你有所帮助还请三连。