一,什么是异常
异常是在程序中导致程序中断运行的一种指令流。
二,异常体系结构
1,异常指的是Exception , Exception类, 在Java中存在一个父类Throwable(可能的抛出) Throwable存在两个子类:
- 1.Error:表示的是错误,是JVM发出的错误操作,只能尽量避免,无法用代码处理。(比如内存过小)
- 2.Exception:一般表示所有程序中的错误,所以一般在程序中将进行try…catch的处理。
- 3.受检异常:受检查的异常,写出来的代码飘红。
- 4.非受检异常:不受检查的异常,运行结果发生问题。
2,多异常捕获的注意点:
- 1、 捕获更粗的异常不能放在捕获更细的异常之前。
- 2、 如果为了方便,则可以将所有的异常都使用Exception进行捕(多态)
3,特殊的多异常捕获写法:
cathc(异常类型1 |异常类型2 对项名){
//表示此块用于处理异常类型1和异常类型2 的异常信息
}
三,处理异常
1,如果要想对异常进行处理,则必须采用标准的处理格式,处理格式语法如下:
try{
// 有可能发生异常的代码段
}catch(异常类型1 对象名1){
// 异常的处理操作
}catch(异常类型2 对象名2){
// 异常的处理操作 } ...
finally{
// 异常的统一出口
//无论是否发生异常,finally必然执行。(同理,必然会释放)
}
try+catch的处理流程:
1,一旦产生异常,则系统会自动产生一个异常类的实例化对象。
注:如果创建的异常类对象不处理,就会通过一种类似返回的机制,抛给调用main方法的人(虚拟机),虚拟机发现自己创建的异常未处理便会中断 程序,为了避免JVM接收到异常对象,在返回虚拟机的过程中拦截掉,这种拦截操作叫做捕获异常。
2,那么,此时如果异常发生在try语句,则会自动找到匹配的catch语句执行,如果没有在catch语句中,则会将异常抛出。
3,所有的catch根据方法的参数匹配异常类的实例化对象,如果匹配成功,则表示由该此catch进行处理。
新手误区:
容易想当然:认为所有用户都会按照程序员的思路来操作;
容易“我认为”:用户的需求或设计在别人看来就是合理的,程序员不是产品经理,不要擅自改动;
finally常见问题:
1,即使return,在准备返回值与跳出函数之间,仍会执行finally中的语句
2:
3:建立栈内存a=10,将a等号右边的内容复制备份一份,接着finally将a变成20,但是备份的a=10,所以返回值为10.
4:new person在堆内存中创建一个空间,该空间中有age=18,创建完空间有一内存地址0x123,并将地址给person p,接着备份person p等号右边的内容(地址0x123),最后finally将0x123中的内容age更改为28,备份的是地址,故return p 返回的是0x123的内容,所以age=28.
5:try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?
答:finally中的代码会执行
详解:执行流程:
1. 先计算返回值, 并将返回值存储起来, 等待返回
2. 执行finally代码块
3. 将之前存储的返回值, 返回出去;
需注意:
1. 返回值是在finally运算之前就确定了,并且缓存了,不管finally对该值做任何的改变,返回的值都不 会改变
2. finally代码中不建议包含return,因为程序会在上述的流程中提前退出,也就是说返回的值不是try或 catch中的值
3. 如果在try或catch中停止了JVM,则finally不会执行.例如停电,软件被关闭,电脑关机 ,或通过如下代码退出 System.exit(0);
四, throws
throws关键字,此关键字主要在方法的声明上使 用,表示方法中不处理异常,而抛给调用处处理。
返回值 方法名称()throws Exception{
}
异常是否抛出去:
如果是因为传入的参数导致异常的发生,则可以通过throws抛出异常。通常是谁调用谁处理;
如果是在此方法中调用时,可以使用try-catch处理异常,并使程序正常运行;
五,throw
1,throw关键字表示在程序中人为的抛出一个异常,因为从异常处理机制来看,所有的异常一旦产生之后,实际上抛出 的就是一个异常类的实例化对象,那么此对象也可以由throw直接抛出。(真正应用的时候,自己造异常还是比较麻烦的,之前加判断也可以出现相同效果。所以用的比较少)
代码:throw new Exception(“抛着玩的。”);
通过出错的方式告诉调用者发生了什么错误:新建实例化异常对象
运行:
2,RuntimeExcepion与Exception的区别
注意观察如下方法的源码:
Integer类: public static int parseInt(String text)throws NumberFormatException
此方法抛出了异常, 但是使用时却不需要进行try。。。catch捕获处理,
原因: 因为NumberFormatException并不是Exception的直接子类,而是RuntimeException的子类,只要是 RuntimeException的子类,则表示程序在操作的时候可以不必使用try…catch进行处理,如果有异常发生,则由JVM进 行处理。当然,也可以通过try catch处理
六,自定义异常类
编写一个类, 继承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。
}
};
自定义运行时异常: