• Java 异常(一) 异常概述及其架构


    Java 异常(一) 异常概述及其架构

    一、异常概述

    (一)、概述

    Java异常是Java提供的一种识别及响应错误的一致性机制。异常指的是程序在执行过程中,出现的非正常的情况,最终会导致JVM的非正常停止。
    Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性。在有效使用异常的情况下,异常能清晰的回答 what, where, why 这3个问题:异常类型回答了“什么”被抛出,异常堆栈跟踪回答了“在哪“抛出,异常信息回答了“为什么“会抛出。

    (二)、异常体系

    异常机制其实是帮助我们找到程序中的问题,异常的根类是java.lang.Throwable,其下有两个子类:java.lang.Errorjava.lang.Exception,平常所说的异常指java.lang.Exception

     Throwable: 有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。异常和错误的区别是:异常能被程序本身可以处理,错误是无法处理。

    1、Throwable:Throwable是 Java 语言中所有错误或异常的超类。Throwable包含两个子类: Error 和 Exception。它们通常用于指示发生了异常情况。

    Throwable包含了其线程创建时线程执行堆栈的快照,它提供了printStackTrace()等接口用于获取堆栈跟踪数据等信息。

    Throwable常用API:

    public void printStackTrace() // 打印异常的详细信息。包含了异常的类型,异常的原因,还包括异常出现的位置。
    public String getMessage() // 获取发生异常的原因。
    public String toString() // 获取异常的类型和异常描述信息。

    2、Error:严重错误Error,无法通过处理的错误,只能事先避免。

    3、Exception:表示异常,异常产生后程序员可以通过代码的方式纠正,使程序继续运行,是必须要处理的。

    (三)、异常分类

    我们平常说的异常就是指Exception,因为这类异常一旦出现,我们就要对代码进行更正,修复程序。

    异常主要分为两大类:编译期异常和运行期异常。

    编译期异常:checked异常。在编译期,就会检查,如果没有处理异常,则编译失败(如日期格式化异常)。

    特点: Java编译器会检查它。此类异常,要么通过throws进行声明抛出,要么通过try-catch进行捕获处理,否则不能通过编译。例如,CloneNotSupportedException就属于被检查异常。当通过clone()接口去克隆一个对象,而该对象对应的类没有实现Cloneable接口,就会抛出CloneNotSupportedException异常。被检查异常通常都是可以恢复的。

    运行期异常:runtime异常。在运行时期,检查异常.在编译时期,运行异常不会编译器检测(不报错)(如数学异常)。

    特点: Java编译器不会检查它。也就是说,当程序中可能出现这类异常时,倘若既"没有通过throws声明抛出它",也"没有用try-catch语句捕获它",还是会编译通过。例如,除数为零时产生的ArithmeticException异常,数组越界时产生的IndexOutOfBoundsException异常,fail-fail机制产生的ConcurrentModificationException异常等,都属于运行时异常。

    虽然Java编译器不会检查运行时异常,但是我们也可以通过throws进行声明抛出,也可以通过try-catch对它进行捕获处理。

    如果产生运行时异常,则需要通过修改代码来进行避免。例如,若会发生除数为零的情况,则需要通过代码避免该情况的发生!

    二、异常处理

    Java异常处理机制用到的几个关键字:try、catch、finally、throw、throws

    • try:用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块之内,当try语句块内发生异常时,异常就被抛出。
    • catch:用于捕获异常。catch用来捕获try语句块中发生的异常。
    • finally:finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接停止。
    • throw:用于抛出异常。
    • throws:用在方法签名中,用于声明该方法可能抛出的异常。

    三、异常注意点

    (一)、finally中的异常会覆盖(消灭)前面try或者catch中的异常

    • 不要在fianlly中使用return。
    • 不要在finally中抛出异常。
    • 减轻finally的任务,不要在finally中做一些其它的事情,finally块仅仅用来释放资源是最合适的。
    • 将尽量将所有的return写在函数的最后面,而不是try ... catch ... finally中。

    (二)、多个异常使用捕获

    • 多个异常分别处理。
    • 多个异常一次捕获,多次处理。
    • 多个异常一次捕获一次处理。

    一般情况下,一般我们是使用一次捕获多次处理方式,如下代码

    try{
         // 编写可能会出现异常的代码
    }catch(异常类型A  e){  当try中出现A类型异常,就用该catch来捕获.
         // 处理异常的代码
         //记录日志/打印异常信息/继续抛出异常
    }catch(异常类型B  e){  当try中出现B类型异常,就用该catch来捕获.
         // 处理异常的代码
         //记录日志/打印异常信息/继续抛出异常
    }

    注意:这种异常处理方式,要求多个catch中的异常不能相同,并且若catch中的多个异常之间有子父类异常的关系,那么子类异常要求在上面的catch处理,父类异常在下面的catch处理。

    三、实例

    (一)、try - catch用法

    try - catch 必须搭配使用,不能单独使用。

    public class ExceptionDemo {
        public static void main(String[] args) {
            try {
                int i = 10 / 0;// 除数不能为0,此行会抛出 ArithmeticException 异常
                System.out.println("i = " + i);// 抛出异常后,此行不会执行
            }catch(ArithmeticException e) { // 捕获 ArithmeticException 异常
                // 在catch 代码块处理异常
                e.printStackTrace(); // 异常最详细信息
                System.out.println("e.getMessage() : " + e.getMessage());// 发生异常的原因
                System.out.println("e.toString() : " + e.toString());  // 获取异常的类型和异常描述信息
            }
        }
    }

    (二)、finally 用法

    try - catch - finally搭配使用,或者 try - finally 搭配使用。

    public class ExceptionDemo {
        public static void main(String[] args) {
            // try-catch-finally搭配使用
            try {
                int[] arr = {1,2,3};
                int i = arr[3];// 数组索引越界,此行会抛出 ArrayIndexOutOfBoundsException 异常
                System.out.println("i = " + i);// 抛出异常后,此行不会执行
            }catch(ArithmeticException e) { // 捕获 ArithmeticException
                System.out.println(e.getMessage());// 发生异常的原因
                System.exit(0); // 程序强制退出,finally 代码块不会执行
            }finally {// 除了程序强制退出,如(System。exit(0)),无论是否发生异常,finally 代码块总会执行
                System.out.println("this is finally");
            }
            
            // try-finally搭配使用
            try {
                int[] arr = {1,2,3};
                int i = arr[3];// 数组索引越界,此行会抛出 ArrayIndexOutOfBoundsException 异常
                System.out.println("i = " + i);// 抛出异常后,此行不会执行
            }finally { 
                // 无论是否发生异常,finally 代码块总会执行
                System.out.println("this is finally");
            }
        }
    }

    注意点:

    • try-catch-finally 搭配:这种形式捕获异常时,开发者可以在 catch 代码块中处理异常(如打印日志、日志记录等),异常处理权在开发者。
    • try-finally 搭配:这种形式捕获异常时,默认抛出给 JVM 处理,JVM默认处理时调用 e.printStackTrace() 方法打印异常详细信息。
    • finally 代码块:除非程序强制退出,否则无论程序是否发生异常,finally 代码块总会执行。
    • finally 中抛出异常会覆盖(消灭)前面 try 或者 catch 中的异常,尽量避免在 finally 代码块中抛出异常。
    • 如果 try 中和 finally 中都有 return 语句,程序会先执行 finally 中的 return 语句,然后程序块运行结束,而不会执行 try 中的 return 语句。所以尽量不在finally中使用 return 语句。

    (三)、throw 用法

    throw 是用于抛出异常,将这个异常对象传递到调用者处,并结束当前方法的执行

    public static void main(String[] args) {
      try {
        int i = 10 / 0;
        System.out.println("i = " + i);
      }catch(ArithmeticException e) { 
        // 抛出异常,传递自定义异常信息提示
        // 默认抛出给 JVM 处理打印异常详细信息    
        throw new ArithmeticException("除数不能为0");
      } 
    }

    (四)、throws 用法

    throws运用于方法声明之上,用于表示当前方法不处理异常,而是提醒该方法的调用者来处理异常(抛出异常)。

    public class ExceptionDemo {
        public static void main(String[] args) {
            demo();
        }
        public static void demo() throws ArrayIndexOutOfBoundsException{
            try {
                int[] arr = {1,2,3};
                int i = arr[3];
                System.out.println("i = " + i);
            }catch(ArrayIndexOutOfBoundsException e) { 
               System.out.println(e.toString());
            } 
        }
    }
  • 相关阅读:
    centos7 rc.local 开机不执行
    springboot与tomcat中GZip压缩
    使用Nginx对Websocket进行反向代理
    spring-data-redis 关于订阅客户端不断创建新线程的问题
    使用apache.tika判断文件类型
    简单理解TCP通信的三次握手
    logback异步输出日志(生产者消费者模型),并非批量写入日志。
    将文本(lrc,txt)文件转换成UTF-8格式
    docker入门(三)
    Spring session(redis存储方式)监听导致创建大量redisMessageListenerContailner-X线程
  • 原文地址:https://www.cnblogs.com/lingq/p/12942895.html
Copyright © 2020-2023  润新知