异常处理 — 学习笔记
1.什么是异常?
异常是在程序中导致程序中断运行的一种指令流。
打个比方,如果代码有300行,不过运行代码时忽然在第6行代码出错了,位置处产生异常,一旦产生异常后,那么第6行代码的语句将不再执行了,所以现在的程序并没有正确的执行完毕之后就退出了。为了保证程序出现异常之后仍然可以正确的执行完毕,所以要采用异常的处理机制。
2.异常体系结构
异常指的是Exception , Exception类, 在Java中存在一个父类Throwable(可能的抛出)
Throwable存在两个子类:
- 1.Error:表示的是错误,是JVM发出的错误操作,只能尽量避免,无法用代码处理。
- 2.Exception:一般表示所有程序中的错误,所以一般在程序中将进行try…catch的处理。
RuntimeExcepion与Exception的区别
多异常捕获的注意点: - 1、 捕获更粗的异常不能放在捕获更细的异常之前。
- 2、 如果为了方便,则可以将所有的异常都使用Exception进行捕获。
特殊的多异常捕获写法:
try{
//可能发生异常的地方
}catch(异常类型1 | 异常类型2 对象名){
//表示此块用于处理异常类型1 和 异长类型2 的异常信息
}
3.处理异常
如果出现了异常,那么处理时必须采用标准的处理格式。处理格式语法如下:
try{
//可能发生异常的代码段
}catch(异常类型1 对象名1){
//异常的处理操作
}catch(异常类型2 对象名2){
//异常处理的操作
}//...其他catch块和异常处理操作
finally{
//异常的统一出口
}
异常的产生流程
- 1、 一旦产生异常,则系统会自动产生一个异常类的实例化对象。
- 2、 那么,此时如果异常发生在try语句,则会自动找到匹配的catch语句执行,如果没有在try语句中,则会将异
常抛出. - 3、 所有的catch根据方法的参数匹配异常类的实例化对象,如果匹配成功,则表示由此catch进行处理。
try+catch的处理流程
- 发生了异常(JVM 根据异常的情况,创建了一个异常对象 - 包含了一异常信息)
- 入口Main 未处理,自动把异常抛给了main 的调用者JVM
- JVM对异常信息进行了相应(将异常信息显示到控制台,然后中断处理)
finally
在进行异常的处理之后,在异常的处理格式中还有一个finally语句,那么此语句将作为异常的统一出口,不管是否产生了异常,最终都要执行此段代码。
throws关键字
在程序中异常的基本处理已经掌握了,但是随异常一起的还有一个称为throws关键字,此关键字主要在方法的声明上使用,表示方法中不处理异常,而交给调用处处理。
格式:
返回值 方法名称()throws Exception{
}
throw关键字
throw关键字表示在程序中人为的抛出一个异常,因为从异常处理机制来看,所有的异常一旦产生之后,实际上抛出的就是一个异常类的实例化对象,那么此对象也可以由throw直接抛出。
格式:
throw new Exception("显示错误信息");
//就是通过人为操作产生新的异常
自定义异常类 了解
编写一个类, 继承Exception,并重写一参构造方法 即可完成自定义受检异常类型。编写一个类, 继承RuntimeExcepion,并重写一参构造方法 即可完成自定义运行时异常类型。
例如:
class MyException extends Exception{ // 继承Exception,表示一个自定义异常类
public MyException(String msg){
super(msg) ; // 调用Exception中有一个参数的构造
}
}
自定义异常可以做很多事情, 例如:
class MyException extends Exception{
public MyException(String msg){
super(msg) ;
//在这里给维护人员发短信或邮件, 告知程序出现了BUG。
}
}
异常处理常见面试题
- try-catch-finally 中哪个部分可以省略?
答:catch和finally可以省略其中一个,catch和finally不能同时省略
注意:格式上允许省略catch块, 但是发生异常时就不会捕获异常了,我们在开发中也不会这样去写代码。 - try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?
答:finally中的代码会执行
详解:
执行流程:
① 先计算返回值, 并将返回值存储起来, 等待返回
② 执行finally代码块
③ 将之前存储的返回值, 返回出去;
需注意:
– 1. 返回值是在finally运算之前就确定了,并且缓存了,不管finally对该值做任何的改变,返回的值都不会改变。
– 2. finally代码中不建议包含return,因为程序会在上述的流程中提前退出,也就是说返回的值不是try或catch中的值。
– 3.如果在try或catch中停止了JVM,则finally不会执行.例如停电等一系列强制中断程序的操作, 或通过如下代码退出JVM:System.exit(0);(0表示正常退出,1和2均为非正常退出)。 - final、finally、finalize的区别?
(1)、final用于声明变量、方法和类的,分别表示变量值不可变,方法不可覆盖,类不可以继承
(2)、finally是异常处理中的一个关键字,表示finally{}里面的代码一定要执行
(3)、finalize是Object类的一个方法,在垃圾回收的时候会调用被回收对象的此方法。 - 你平时在项目中是怎样对异常进行处理的。
1)尽量避免出现runtimeException 。例如对于可能出现空指针的代码,带使用对象之前一定要判断一下该对象是否为空,必要的时候对runtimeException也进行try catch处理。
(2)进行try catch处理的时候要在catch代码块中对异常信息进行记录,通过调用异常类的相关方法获取到异常的相关信息,返回到web端,不仅要给用户良好的用户体验,也要能帮助程序员良好的定位异常出现的位置及原因。例如,以前做的一个项目,程序遇到异常页面会显示一个图片告诉用户哪些操作导致程序出现了什么异常,同时图片上有一个按钮用来点击展示异常的详细信息给程序员看的。 - 运行时异常和一般异常有何不同?
- Java提供了两类主要的异常:runtimeException和checkedException
一般异常(checkedException)主要是指IO异常、SQL异常等。对于这种异常,JVM要求我们必须对其进行cathc处理,所以,面对这种异常,不管我们是否愿意,都是要写一大堆的catch块去处理可能出现的异常。
运行时异常(runtimeException)我们一般不处理,当出现这类异常的时候程序会由虚拟机接管。比如,我们从来没有去处理过NullPointerException,而且这个异常还是最常见的异常之一。
出现运行时异常的时候,程序会将异常一直向上抛,一直抛到遇到处理代码,如果没有catch块进行处理,到了最上层,如果是多线程就有Thread.run()抛出,如果不是多线程那么就由main.run()抛出。抛出之后,如果是线程,那么该线程也就终止了,如果是主程序,那么该程序也就终止了。