一、异常的概述
1.概述
异常就是Java程序在运行过程中出现的错误。
Throwable
类是 Java 语言中所有错误或异常的超类。
只有当对象是此类(或其子类之一)的实例时,才能通过 Java 虚拟机或者 Java throw
语句抛出。
类似地,只有此类或其子类之一才可以是 catch
子句中的参数类型。
2.分类
两个子类的实例,Error
和 Exception
,通常用于指示发生了异常情况。
3.继承体系
Throwable父类中已经实现了异常所需要的所有功能,所以Exception类、Error类以及两者众多子类的区别只是取了不同名字加以区分,并且具有各自的构造函数,构造函数中的参数全部传递给父类,最终传递给Throwable。
例如Exception类中:
public Exception() {
super();
}
public Exception(String message) {
super(message);
}
public Exception(String message, Throwable cause) {
super(message, cause);
}
public Exception(Throwable cause) {
super(cause);
}
二、如何处理异常
函数接受到异常时有两种处理方式
- 函数内部处理该问题,然后继续执行
- 函数内部无处理方式,将该问题提交给调用该函数的jvm处理,jvm使用默认的异常处理机制进行处理,并将该异常的名称、信息、出现位置输出在控制台上,然后停止执行
三、 编译期异常和运行期异常
Java中的异常被分为两大类:编译时异常和运行时异常。
所有的RuntimeException类及其子类的实例被称为运行时异常,其他的异常就是编译时异常
- 编译时异常
Java程序必须显示处理,否则程序就会发生错误,无法通过编译;在编译某个程序的时候,有可能会有这样那样的事情发生,比如文件找不到,这样的异常就必须在编译的时候处理 如果不处理编译通不过
- 运行时异常
无需显示处理,也可以和编译时异常一样处理,如果未处理,编译可以通过,但是在运行时发生错误;就是程序员所犯得错误,需要回来修改代码
四、 异常处理
1.异常处理的两种方式
- try...catch
- throws
2.try...catch处理异常
①基本格式:
- try catch
- try catch finally
- try finally
②try...catch的方式处理一个异常
public static void main(String[] args) {
try{
int x=10/0;
System.out.println(x);
}catch(ArithmeticException a) { //ArithmeticException a = new ArithmeticException();
System.out.println("出错了,除数为零了");
}
}
③try...catch的方式处理多个异常
public static void demo1() {
int a = 10;
int b = 0;
int[] arr = {11,22,33,44,55};
try {
System.out.println(a / b);
System.out.println(arr[10]);
arr = null;
System.out.println(arr[0]);
} catch (ArithmeticException e) {
System.out.println("除数不能为零");
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("索引越界了");
} catch (Exception e) { //Exception e = new NullPointerException();
System.out.println("出错了");
}
System.out.println("over");
}
int a = 10;
int b = 0;
int[] arr = {11,22,33,44,55};
//JDK7如何处理多个异常
try {
System.out.println(a / b);
System.out.println(arr[10]);
} catch (ArithmeticException | ArrayIndexOutOfBoundsException e) {
System.out.println("出错了");
}
注:
-
try后面如果跟多个catch,那么小的异常放前面,大的异常放后面,根据多态的原理,如果大的放前面,就会将所有的子类对象接收,后面的catch就没有意义了
-
安卓,客户端开发,一般try{}catch(Exception e){}
EE,服务端开发,一般都是底层开发,从底层向上抛
3. throws的方式处理异常
定义功能方法时,需要把出现的问题暴露出来让调用者去处理,那么就通过throws在方法上标识。
编译时异常的抛出必须对其进行处理
运行时异常的抛出可以处理也可以不处理
public void setAge(int age) throws AgeOutOfBoundsException {
if(age >0 && age <= 150) {
this.age = age;
}else {
//Exception e = new Exception("年龄非法");
//throw e;
throw new AgeOutOfBoundsException("年龄非法");
}
}
public void setAge(int age){ //运行时异常可以不处理
if(age >0 && age <= 150) {
this.age = age;
}else {
throw new RuntimeException("年龄非法");
}
}
五、throw的概述以及和throws的区别
1.概述
在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。
2.throws和throw的区别
用在方法声明后面,跟的是异常类名
可以跟多个异常类名,用逗号隔开
表示抛出异常,由该方法的调用者来处理
用在方法体内,跟的是异常对象名
只能抛出一个异常对象名
表示抛出异常,由方法体内的语句处理
- throws
- throw
六、Throwable的几个常见方法
String
getMessage()
返回此 throwable 的详细消息字符串。
String
toString()
返回此 throwable 的简短描述。
void
printStackTrace()
将此 throwable 及其追踪输出至标准错误流。
jvm默认使用第三种方式处理异常
七、finally关键字的特点及作用
1.finally的特点
- 被finally控制的语句体一定会执行
- 特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))
-
return语句在执行之前,会看一看finally中有没有需要执行的程序,如果有就将finally执行后在彻底返回
2.finally的作用
- 用于释放资源,在IO流操作和数据库操作中会见到
3.面试题
①如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后。
public int method() {
int x = 10;
try {
x = 20;
System.out.println(1/0);
return x;
} catch (Exception e) {
x = 30;
return x;
} finally {
x = 40;
//return x;
//千万不要在finally里面写返回语句,因为finally的作用是为了释放资源,是肯定会执行的
// 如果在这里面写返回语句,那么try和catch的结果都会被改变,所以这么写就是犯罪
}
}
在return前执行,此时已经将x的值置入返回通道,在return彻底执行之前会执行finally中的语句,x的值会改变,但是通道中的值不会改变,所以最后x=30
②final,finally和finalize的区别
final
可以修饰类,不能被继承
修饰方法,不能被重写
修饰变量,只能赋值一次
finally
try语句中的一个语句体,不能单独使用,用来释放资源
finalize
是一个方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
八、自定义异常的基本使用
class AgeOutOfBoundsException extends Exception {
public AgeOutOfBoundsException() {
super();
}
public AgeOutOfBoundsException(String message) {
super(message);
}
}
九、异常的注意事项
1.注意事项
- 子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。(父亲坏了,儿子不能比父亲更坏)
- 如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者是他的子集,子类不能抛出父类没有的异常
- 如果被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常,如果子类方法内有异常发生,那么子类只能try,不能throws
2.如何使用
原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,这是用throws
区别:
- 后续程序需要继续运行就try
- 后续程序不需要继续运行就throws
如果JDK没有提供对应的异常,需要自定义异常。