文章目录
12.1 问题
.....
public static void main(String[] args) {
int num1 = 10;
int num2 = 0;
int res = num1 / num2;
System.out.println("程序继续运行");
}
因除数不能为0,所以上面代码会报错,但是若因为这个不太重要的问题就退出程序,那么代码健壮性就太差 - - 引出异常
12.2 解决方案 - 异常捕获
对异常进行捕获,保证程序可以继续运行
.....
public static void main(String[] args) {
int num1 = 10;
int num2 = 0;
try {
int res = num1 / num2;
} catch(Exception e) {
System.out.println("出现异常原因 = " + e.getMessage());
}
System.out.println("程序继续运行");
}
12.3 异常
概念:Java语言中,将程序执行中发生的不正常情况称为“异常”。
注意:开发中的语法错误和逻辑错误不是异常
执行过程中所发生的异常事件可分为两大类:
- Error(错误):java虚拟机无法解决的严重问题。如:jvm系统内部错误、资源耗尽等严重情况。如:StackOverflowError(栈溢出)和OOM(out of memory),Error是严重错误,程序会崩溃。
- Exception:其他因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。例如空指针访问,试图读取不存在的文件,网络连接中断等等,Exception分为两大类:运行时异常(程序运行时,发生的异常)和 编译时异常(编程时,编译器检查出的异常)
12.4 异常体系图一览
12.4.1 异常体系图
12.4.2 小结
- 异常分为两大类,运行时异常和编译时异常
- 运行时异常,编译器检查不出来。一般是指编程时的逻辑错误,是程序员应该避免其出现的异常。java.lang.RuntimeException类及它的子类都是运行时异常。
- 对于运行时异常,可以不作处理,因为很普遍,若全部处理可能会对程序的可读性和运行效率产生影响
- 编译时异常,是编译器要求必须处置的异常
12.5 常见的运行时异常
12.5.1 五种常见运行时异常
- NullPointerException空指针异常
- ArithmeticException数学运算异常
- ArrayIndexOutOfBoundsException数组下标越界异常
- ClassCastException类型转换异常
- NumberFormatException数字格式不正确异常
12.5.2 举例
- 空指针异常
.....
String name = null;
System.out.println(name.length());
.....
- 数学运算异常
如除数为0 - 数组下标越界异常
.....
int[] arr = {1, 2, 3};
for(int i = 0; i <= arr.length; i++) {
System.out.println(arr[i]);
}
- 类型转换异常
试图将对象强制转换为不是实例的子类时,抛出该异常
.....
public static void main(String[] args) {
A b = new B();//向上转型
B b2 = (B)b;//向下转型
C c2 = (C)b;//异常
}
class A {}
class B extends A {}
class C extends A {}
- 数字格式不正确异常
当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出异常。
如:将 name = “jack”转化成数值num
int num = Integer.parseInt(name);
12.6 编译异常
12.6.1 介绍
编译异常是指在编译期间,就必须处理的异常,否则代码不能通过编译
12.6.2 常见的编译异常
12.7 异常处理
12.7.1 基本介绍
异常处理就是当异常发生时,对异常处理的方式
12.7.2 两种异常处理方式
- try - catch - finally 程序员在代码中捕获发生的异常,自行处理
- throws 将发生的异常抛出,交给调用者(方法)来处理,最顶级的处理者就是JVM
12.7.3 示意图
12.8 try - catch 异常处理
12.8.1 处理异常说明
- Java提供try和catch块来处理异常。try块用于包含可能出错的代码,catch块用于处理try块中发生的异常。可以根据需要在程序中有多个try…catch块
- 基本语法
try {
//可疑代码
//将异常生成对应的异常对象,传递给catch块
} catch(异常) {
//对异常的处理
}
//如果没有finally,语法是可以通过
12.8.2 快速入门
.....
public static void main(String[] args) {
int num1 = 10;
int num2 = 0;
try {
int res = num1 / num2;
} catch(Exception e) {
System.out.println(e.getMessage());
}
}
12.8.3 注意事项
- 如果发生异常,则异常发生后面的代码不会执行,直接进入到catch块
- 如果没发生异常,则顺序执行try代码块,不会进入catch
- 如果希望不管是否发生异常,都执行某段代码(比如关闭连接,释放资源等)使用finally
- 可以有多个catch语句,捕获不同的异常(进行不同的业务处理),要求父类异常在后,子类异常在前,如果发生异常只会匹配一个catch
- 可以进行try - finally 配合使用,这种用法相当于没有捕获异常,因此程序会直接崩掉/退出。应用场景:执行一段代码,不管是否发生异常,都必须执行某个业务逻辑
12.9 throws 异常处理
12.9.1 基本介绍
- 如果有一个方法中的语句执行时可能生成某种异常,但是并不确定如何处理这种异常,则此方法应显示的声明抛出异常,表面该方法将不对这些异常进行处理,而由该方法的调用者负责处理
- 在方法声明中用throws语句可以声明抛出异常的列表,throws后面的异常类型可以是方法中产生的异常类型,也可以是它的父类
12.9.2 快速入门
.....
public static void readFile(String file) throws FileNotFoundException {
..
//读文件的操作可能产生FileNotFoundException类型的异常
FileInputStream fis = new FileInputStream("d://aa.txt");
..
}
.....
12.9.3 注意事项和使用细节
- 对于编译异常,程序中必须处理
- 对于运行时异常,程序中如果没有处理,默认就是throws的方式处理
- 子类重写父类的方法时,对抛出异常的规定:子类重写的方法,所抛出的异常类型要么和父类抛出的异常一致,要么为父类抛出异常的类型的子类型
- 在throws过程中,如果有方法try - catch,就相当于处理异常,就可以不必throws
12.10 自定义异常
12.10.1 基本概念
当程序中出现了某些“错误”,但该错误信息并没有在Throwable子类中描述处理,这个时候可以自己设计异常类,用于描述该错误信息。
12.10.2 自定义异常的步骤
- 定义类:自定义异常类名 继承 Exception或RuntimeException
- 如果继承Exception,属于编译异常
- 如果继承RuntimeException,属于运行异常(推荐)
12.10.3 应用实例
public class CustomException {
public static void main(String[] args) {
int age = 80;
if(!(age >= 80 && age <= 120)) {
//通过构造器,设置信息
throw new AgeException("年龄需要在80 - 120");
}
System.out.println("你的年龄范围正确");
}
}
class AgeException extends RuntimeException {
public AgeException(String message) {
super(message);
}
}