异常知识点总结

异常

什么是异常

简单来说异常就是用来表示Java程序运行过程中的错误(信息)

网络中断,用户输入信息,读取不存在的文件

异常体系与分类

在这里插入图片描述

在Java中Throwable作为所有错误跟异常的祖先类

根据错误的严重程度分

  • Error:比较严重的错误(代码处理不了, Jvm内部资源耗尽的错误)

    • java.lang.StackOverflowError 栈溢出

    • java.lang.OutOfMemoryError 堆溢出

  • Exception:错误程度小, 能够用代码进行处理

    • java.lang.ArithmeticException: / by zero 算数异常 /0
    • java.lang.NullPointerException 空指针异常
    • java.lang.ArrayIndexOutOfBoundsException 数组越界

根据处理方式的不同

  • 编译时异常(checked) : 编译不通过(除了RuntimeException及其子类外的其他的异常)
  • 运行时异常(unchecked): 编译通过, 但是运行时可能会出错(RuntimeException及其子类)

需要注意的是:Exception是编译时异常和运行时异常的父类,在自定义异常的时候,Exception作为编译时异常。

常见异常

在这里插入图片描述

常见异常
    编译时异常:
        java.lang.CloneNotSupportedException 克隆异常
        java.io.FileNotFoundException 找不到文件
        java.io.IOException IO 操作异常     
    运行时异常:
        java.lang.ArithmeticException 算数异常
        java.lang.NullPointerException 空指针异常
        java.lang.ArrayIndexOutOfBoundsException 数组越界异常
        java.lang.NumberFormatException 数字格式化异常
        java.lang.ClassCastException 类型转换异常

异常处理

jvm默认处理机制

jvm默认异常处理流程

  1. 当我们代码在执行到,发生错误的地方。
  2. 一旦发生错误,jvm就会终止我们自己程序的运行,转而执行jvm自己的错误处理流程
  3. 在发生错误地方,收集错误信息,产生一个描述错误的对象
  4. 访问收集到的错误信息,将错误信息,输出到控制台窗口中(哪个线程,异常类型名, 异常原因, 哪个类哪个方法哪一行报错了)

执行过程

  • 如果错误产生在main方法中
    • 当我们的代码执行到错误行数之前,代码是正常执行的
    • 当我们的代码执行到错误行数时,JVM会终止程序的执行,抛出一个该异常信息封装成的对象
    • 将该对象中的异常信息,打印到控制台上,告诉程序员发生了什么问题
    • 发生错误之后的语句,都不执行了
  • 如果错误产生在main方法当中的另一个方法中
    • 当程序执行到该方法的错误行数时,JVM会终止程序的执行
      • 向上给方法的调用者抛出一个该异常信息封装成的对象
    • 一直向上抛出,直到抛给main方法,main方法最终抛给JVM
    • 发生异常之前的语句正常执行,但是之后的语句都不执行了
  • 默认处理机制仅针对运行时异常

捕获异常,自己处理

try-catch

Java中对异常有针对性的语句进行捕获,可以对出现的异常进行指定方式的处理。

单分支

语法

方式一:
try{
   // 可能出现异常的代码 
}catch(异常类型 对象名){
   // 对异常的处理操作 
}
方式二:
try{
   // 可能出现异常的代码 
}catch(异常类型1 | 异常类型2 | 异常类型3 | 对象名){
   // 对异常的处理操作 
}

try-catch的执行:

  1. 如果try中代码运行时发生了错误,jvm在发生错误的代码处,收集错误信息
  2. try 块中在错误代码之后的代码,就不会在运行,jvm会跳转到相应的错误处理器中,
    执行开发者自己写的,错误处理代码
  3. 错误处理器中的代码,一旦执行完毕紧接着,程序继续正常执行,执行的是整个try代码块之后的代码

注意:catch代码块中的代码,只有try块中的代码执行出错时,并且类型匹配成功才会执行!

快捷键:选中代码片段,Ctrl + Alt + t

捕获异常信息
//获取异常信息,返回字符串。
getMessage()
//获取异常类名和异常信息,返回字符串。
toString()
//获取异常类名和异常信息,以及异常出现在程序中的位置,并打印到控制台
printStackTrace()
多分支

语法

try{
   // 可能出现异常的代码 
}catch(异常类型 对象名){
   // 对异常的处理操作 
}catch(异常类型 对象名){
   // 对异常的处理操作 
}catch(异常类型 对象名){
   // 对异常的处理操作 
}.....

匹配规则:

  1. 根据实际的异常对象的类型,和异常分支(异常处理器)声明的异常类型,从上到下一次做类型匹配
  2. 一旦通过类型匹配,发现实际异常对象的类型和Catch分支(异常处理器)声明的异常类型,类型匹配,就把异常对象交给这个异常分支(异常处理器)
  3. 多分支的异常处理的执行,有点类似于多分支if-else的执行,一次匹配,只会执行多个catch分支中的一个

注意事项:

如果说,在多catch分支的情况下,如果不同的catch分支,处理的异常类型,有父子关系,那么就一定要注意,处理子类的异常分支写在前面,父类的异常分支写在后面来兜底。

抛出异常,上层处理

throws关键字

在方法定义时使用,声明该方法可能抛出的异常,这是对于编译时异常的,可以在语法层面强制方法调用者处理该异常

基本语法:

修饰符 返回值 方法名(形参列表) throws 异常列表 {}

解释说明:

  • 异常列表: 异常类型1, 异常类型2, … 用逗号隔开,列表中的异常不要出现父子关系,如果有,那么编译器只会强制处理父类
  • 只是声明可能抛出,到底抛不抛,看代码
  • throws+运行时异常没有意义,因为运行时异常会自动抛出,不需要声明

子类重写父类方法注意:

子类方法不能比父类抛出的编译时异常范围更大的编译时异常。父类如果抛出Exception,那么子类抛不抛出已经无所谓了。所以建议子类重写的时候保持跟父类一样的异常列表。

public class Demo {
    public static void main(String[] args) {
        //m1();
        //m2(); // 执行结果与m1一样
        try {
            m3();
        }catch (CloneNotSupportedException e) {
            System.out.println("出现了克隆异常");
        }

    }
    // 运行时异常(不加throws)
    private static void m1() {
        System.out.println(10 / 0);
    }
    // 运行时异常(加throws)
    private static void m2() throws ArithmeticException{
        System.out.println(10 / 0);
    }
    // 编译时异常,一定要加throws,表示这个异常不在本方法中处理,交给方法的调用者处理
    private static void m3() throws CloneNotSupportedException{
        Demo demo = new Demo();
        // 如果在没有实现Cloneable接口的实例上调用 Object 的 clone 方法,则会导致抛出 CloneNotSupportedException 异常。
        demo.clone();
    }
}

throw关键字

在方法体中使用,主动在程序中抛出异常,每次只能抛出确定的某个异常对象

基本语法:

throw 异常对象    (new 出来的)

throw可以加运行时异常对象,也可以加编译时异常对象,throw+编译时异常时,需要结合throws关键字

public class Demo {
    public static void main(String[] args) {
        int[] arr1 = null;
        int[] arr2 = {};
        int[] arr3 = {1, 2, 3};
        // test(arr1);
        // test(arr2);
        test(arr3);
    }
    // 对数组遍历的时候加上判断是否出现异常的操作
    public static void test(int[] arr) {
        // 引用为空,抛出异常
        if(arr == null) {
            throw new NullPointerException("出现了空指针异常");
        }
        // 数组长度为0,抛出异常
        if(arr.length == 0) {
            throw new RuntimeException("数组长度为0,不合法");
        }
        // 遍历数组
        for (int i = 0; i < arr.length; i++) {
            System.out.println(arr[i]);
        }
    }
    // 运行时异常
    public void m1() {
        throw new RuntimeException("发生运行时异常");
    }
    // 编译时异常要配合throws使用
    public void m2() throws CloneNotSupportedException{
        throw new CloneNotSupportedException("发生了编译时异常,克隆异常");
    }
}

throws vs throw

throws

  • 用在方法声明后面,跟的是异常类名
  • 可以跟多个异常类名,用逗号隔开
  • 表示抛出异常,由该方法的调用者来处理
  • throws表示出现异常的一种可能性,并不一定会发生这些异常

throw

  • 用在方法体内,跟的是异常对象名
  • 只能抛出一个异常对象
  • 表示抛出异常,可以由方法体内的语句处理
  • throw表示确定抛出了异常,执行throw一定会抛出某种异常

异常策略选择

总结一下,目前为止,我们学习过的异常处理策略有2种:

  • 捕获并处理try-catch
  • 向上抛出
    • 运行时异常,自动抛出,直道抛给jvm
    • 编译时异常,需要结合throws关键字向上抛

如何选择策略?

  • 对于运行时异常,我们不应该写出产生这种异常的代码,应该在代码的测试阶段修正代码。
  • 对于编译时异常,功能内部能够处理的就处理,如果不能够或者没有必要处理,就抛出。

finally

特点

被finally控制的语句体一定会执行
特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))

作用

用于释放资源,在IO流操作和数据库操作中会见到

如何使用

跟try-catch结合

语法:方式一
try{
    
}catch(){
    
}.....
finally{
    // 一定执行
}

方式二:
try{
    
}finally{
    
}

一些细节:

  • try代码块如果有return,程序会先执行完finally代码块,回过头执行try中的return
  • catch代码块中如果有return,并且catch正常捕获异常执行,程序会先执行完finally代码块后,再回去执行catch中return,从catch代码块中结束方法
  • finally代码中有return,不会影响finally代码块执行
  • 如果finally和catch中都有return,程序会直接从finally代码块中的return结束方法
  • 如果try中的异常不能正常捕获,但是finally中有return,注意此时程序会跳过这个异常,不会抛出异常给JVM报错

自定义异常

现有的异常体系不满足需求就需要自定义异常

如何自定义异常

  1. 自定义编译时异常:定义一个类继承Exception,写构造方法

  2. 自定义运行时异常:定义一个类继承RuntimeException,写构造方法

/*
自定义编译时异常
 */
class MyException extends Exception {
    public MyException(String message) {
        super(message);
    }

    public MyException() {
    }
}
/*
自定义运行时异常
 */
class MyRuntimeException extends RuntimeException {
    public MyRuntimeException(String message) {
        super(message);
    }

    public MyRuntimeException() {
    }
}
  • 16
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值