1. 异常的基础知识
-
异常指程序运行中出现的不期而至的各种情况,如:文件找不到,网络连接失败,非法参数等。
-
异常发生在程序运行期间,它影响了正常的程序流程。
-
常见的异常:
package com.exception;
/**
* @author 23862
*/
public class Demo01 {
public static void main(String[] args) {
System.out.println();
/*报错:Exception in thread "main" java.lang.ArithmeticException: / by zero
System.out.println(11/0);*/
}
/*调用时报错:Exception in thread "main" java.lang.StackOverflowError
public void a(){
b();
}
public void b(){
a();
}*/
}
-
异常的简单分类:
- 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在的文件时,异常就发生了,这些异常在编译时不能被简单地忽略。
- 运行时异常:运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
- 错误ERROR:错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译时是检查不到的。
-
异常体系结构:
- Java把异常当做对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类。
- 在Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception。
-
Error
- Error类对象由Java虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关,对于error异常,Java程序是无能为力的。
- Java虚拟机运行错误(Virtual MachineError),当JVM不再有继续执行操作所需的内存资源时,将出现OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止;
- 还有发生在虚拟机试图执行应用时,如类定义错误(NoClassDefFoundError),连接错误(LinkageError)。这些错误是不可查的,因为它们在应用程序的控制和处理能力之外,而且绝大多数是程序运行时不允许出现的状况。
-
Exception
-
Exception分支中有一个重要的子类RuntimeException(运行时异常)
- ArrayIndexOutOfBoundsException(数组下标越界异常)
-
-
NullPointerException(空指针异常)
- ArithmeticException(算数异常)
-
MissingResourceException(丢失资源)
- ClassNotFoundException(找不到类)等异常,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。
-
这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生;
- 非RuntimeException:编译期间就必须处理,否则程序不能通过编译,更不能正常运行
-
Error和Exception的区别:Error通常是灾难性的致命错误,是程序无法控制和处理的,当出现这些异常时,Java虚拟机(Jvm)一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。
2. 异常处理机制
2.1JVM的默认处理方案
如果程序出现了问题,我们没有做任何处理,最终会做默认的处理:
- 把异常的名称,异常原因及异常出现的位置等信息输出在了控制台
- 程序在在出现异常的位置停止执行
2.2异常处理方式
我们自己处理异常有两种方法:
- 抛出异常(throws)
- 捕获异常(try…catch…)
- try…catch…方式
- try…catch…格式:
try {
可能出现异常的代码;
}catch(异常类名 变量名) {
异常的处理代码;
}
执行流程:程序从try内的代码开始执行,当出现异常时会自动生成一个异常对象提交给Java运行时系统,当Java运行时系统接收到异常对象时回到catch内的代码中寻找匹配的异常类,找到后进行相应处理,当流程执行完毕后,程序还会继续往下执行。
- 异常处理五个关键字
- try:监控区域
- catch:捕获异常
- finally:处理善后操作
- throw:在方法内捕获异常
- throws:在方法上捕获异常
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
method();//你访问的数组的索引不存在
System.out.println("结束");//结束
}
public static void method() {
try {
int[] arr = {1,2,3};
//数组下标越界
System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
} catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
System.out.println("你访问的数组的索引不存在");
e.printStackTrace();//该代码会输出JVM的默认处理方案
}
}
}
- 程序出现异常,JVM也完成默认的处理,而且还正常的执行了异常发生之后的代码,进行try…catch操作最主要的目的就是为了让程序出现异常后能正常执行下去。
- throws方式
-
不是所有异常我们都有权限处理,当遇到无法处理的异常时,需要使用Java提供的throws处理方案
-
throws格式:
-
throws 异常类名;
-
该格式是跟在方法的括号后面的
-
编译时异常必须要进行处理,两种处理方案:try…catch…或者throws,如果抛出异常,将来谁调用有异常的方法谁就需要去处理异常。
-
运行时异常可以不处理,出现问题后,需要再次修改代码
-
测试类:
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
method();
System.out.println("结束");
}
//运行时异常
public static void method() throws ArrayIndexOutOfBoundsException {
int[] arr = {1, 2, 3};
System.out.println(arr[3]);
}
}
- 控制台输出“开始”,异常的类名,原因和位置。“结束”没有被输出。说明用抛出异常方式处理运行时异常不是真正的处理,如果想让程序继续往下执行,还是需要try…catch…去处理。
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
try{
method();
} catch (ParseException e){
e.printStackTrace();
}
System.out.println("结束");//结束
}
//编译时异常
public static void method() throws ParseException {
String s = "2022-05-09";
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date d = sdf.parse(s);
System.out.println(d);
}
}
- 控制台输出"开始",“Sun Aug o9 00:00:00 CST 2022”,“结束”。程序发生异常后成功的往下执行了。
3. Throwable的成员方法
方法名 | 说明 |
---|---|
public String getMessage() | 返回此throwable的详细消息字符串 |
public String toString() | 返回此可抛出的简短描述 |
public void printStackTrace() | 把异常的错误信息输出在控制台 |
- public String getMessage()方法:
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
method();
System.out.println("结束");//结束
}
public static void method() {
try {
int[] arr = {1,2,3};
//数组下标越界
System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
} catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
System.out.println(e.getMessage());//方法返回出现异常的原因,Index 3 out bounds for length 3
}
}
}
- public String toString()方法:
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
method();
System.out.println("结束");//结束
}
public static void method() {
try {
int[] arr = {1,2,3};
//数组下标越界
System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
} catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
System.out.println(e.toString());//方法返回出现异常的原因和异常类名,包含了getMessage方法输出的内容。输出:java.lang.ArrayIndexOutOfBoundsException: Index 3 out of bounds for length 3
}
}
}
- public void printStackTrace()方法:
public class ExceptionDemo {
public static void main(String[] args) {
System.out.println("开始");//开始
method();//你访问的数组的索引不存在
System.out.println("结束");//结束
}
public static void method() {
try {
int[] arr = {1,2,3};
//数组下标越界
System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
} catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
System.out.println("你访问的数组的索引不存在");
e.printStackTrace();//该代码会输出JVM的默认处理方案在控制台
}
}
}
4. 编译时异常和运行时异常的区别
Java中的异常被分为两大类:编译时异常和运行时异常,也被称为受检异常和非受检异常
所有的RumtimeException类及其子类被称为运行时异常,其他的异常都是编译时异常
- 编译时异常:必须显示处理(在编写代码时就要处理),否则程序会发生错误,无法通过编译
- 运行时异常:无需显示处理,可以和编译时异常一样处理,一般通过try…catch…处理
public class ExceptionDemo {
public static void main(String[] args) {
method();
}
//运行时异常
public static void method() {
try {
int[] arr = {1, 2, 3};
System.out.println(arr[3]);//在控制台报错ArrayIndexOutOfBoundsException
}catch (ArrayIndexOutOfBoundsException e) {
e.printStackTrace();
}
}
}
public class ExceptionDemo {
public static void main(String[] args) {
method();
}
//编译时异常
public static void method() {
try {
String s = "2022-05-08";
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date d = sdf.parse(s);//因为parse方法可能会出问题所以必须进行处理,捕获和抛出异常都可以
System.out.println(d);
}catch (ParseException e) {
e.printStackTrace();
}
}
}
5. 自定义异常
5.1 使用实例
-
使用Java内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需要继承Exception类即可。
-
格式:
-
public class 异常类名 extends Exception { 无参构造 带参构造 }
-
范例:
-
public class ScoreException extends Exception { public ScoreException() {} public ScoreException(String message) { super(message); } }
-
在程序中使用自定义异常类,大体可分为一下几个步骤:
- 创建自定义异常类。
- 在方法中通过throw关键字抛出异常对象。
- 如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
- 在出现异常方法的调用者中捕获并处理异常。
-
自定义异常类ScoreException:
public class ScoreException extends Exception {
public ScoreException() {}
public ScoreException(String message) {
super(message);
}
}
- Teacher类:
public class Teacher {
public void checkScore(int score) throws ScoreException {//抛出异常类
if(score<0 || score>100) {
throw new ScoreException("你给的分数错误,分数应在0到100之间");//抛出自定义的异常对象
} else {
System.out.println("分数正常");
}
}
}
- 测试类TeacherTest:
public class TeacherTest {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
System.out.println("请输入分数:");
int score = sc.nextInt();
Teacher t = new Teacher();
try{
t.checkScore(score); //处理编译时异常
} catch (ScoreException e) {
e.printStackTrace();
}
}
}
-
在控制台输入"90",控制台输出"分数正常"
-
在控制台输入"111",控制台输出异常的名称和出现异常的位置,以及异常的原因"你给的分数错误,分数应在0到100之间"
-
实际应用中的经验总结:
- 处理运行异常时,采用逻辑去合理规避同时辅助try-catch进行处理。
- 在多重catch块后面,可以加上一个catch(Exception)来处理可能会被遗漏的异常。
- 对于不确定的代码,也可以加上try-catch,处理潜在的异常。
- 尽量去处理异常,切忌只是简单地调用printStackTrace()去打印输出。
- 具体如何处理异常,要根据不同的业务需求和异常类型去决定。
- 尽量添加finally语句块去释放占用的资源。
5.2 throws和throw的区别
-
throws
- 用在方法声明后面,跟的是异常类名
- 表示抛出异常,由该方法的调用者来处理
- 表示出现异常的一种可能性,并不一定会发生这些异常
-
throw
- 用在方法体内,跟的是异常对象名
- 表示抛出异常,由方法体内的语句处理
- 执行throw一定抛出了某种异常