Java 异常处理详解(如果想知道Java中有关异常处理的知识点,那么只看这一篇就足够了!)

        前言:异常处理是 Java 编程中非常重要的一部分,它能够有效地捕获和处理程序运行中的错误,提高代码的健壮性和可靠性。本文将深入探讨 Java 中异常的概念、体系结构、抛出、解决方式以及如何自定义异常,并结合代码案例进行详细解释。


✨✨✨这里是秋刀鱼不做梦的BLOG

✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客

先让我们看一下本文的大致内容:

目录

1.异常的概念

1. NullPointerException(空指针异常)

2. ArrayIndexOutOfBoundsException(数组越界异常)

3. ArithmeticException(算术异常)

2.异常的体系结构

        (1)异常的体系结构

        (2)异常的分类

1.编译时异常:

2.运行时异常:

3.异常的抛出

4.异常的解决方式

        (1)异常声明throws

        (2)使用try - catch - finally语句

补充:异常处理流程总结

5.自定义异常


1.异常的概念

        ——在学习Java中的异常之前,先让我们了解一下什么是Java中的异常:

        在 Java 中,异常(Exception)是指程序在执行过程中遇到的不正常情况,这些情况可能导致程序无法继续执行或产生错误的结果。异常可以是 Java 标准库中提供的内置异常类,也可以是开发人员自定义的异常类。

        相信读者在日常的编写代码的时候,或多或少的都遇到过一些异常,只是读者你可能没有太注意它们,比如如下异常:

1. NullPointerException(空指针异常)

空指针异常通常发生在尝试访问对象的属性或调用对象的方法时,而该对象为 null 的情况下。

public class NullPointerExceptionExample {
    public static void main(String[] args) {
        String str = null;
        System.out.println(str.length()); // 这行代码会抛出 NullPointerException
    }
}

2. ArrayIndexOutOfBoundsException(数组越界异常)

数组越界异常发生在尝试访问数组的不存在的索引位置时。

public class ArrayIndexOutOfBoundsExceptionExample {
    public static void main(String[] args) {
        int[] arr = new int[3];
        System.out.println(arr[5]); // 这行代码会抛出 ArrayIndexOutOfBoundsException
    }
}

3. ArithmeticException(算术异常)

算术异常发生在进行除以零的算术运算时。

public class ArithmeticExceptionExample {
    public static void main(String[] args) {
        int a = 10;
        int b = 0;
        System.out.println(a / b); // 这行代码会抛出 ArithmeticException
    }
}

        当然,这以上只是举出来几个常见的异常的案例,所以我们就可以知道了异常类型和如何处理它们对于编写健壮的 Java 代码非常重要。

2.异常的体系结构

        

        (1)异常的体系结构

        在Java中,Java 异常体系结构是由一组类构成的,它们都是 Throwable 类的子类。常见的异常类包括 ErrorRuntimeException 以及它们的子类。Error 类及其子类通常表示严重的错误,如系统错误或虚拟机错误,程序通常无法恢复。RuntimeException 及其子类通常表示程序逻辑错误或运行时错误,如空指针异常、数组越界异常等。

我们可以直接看一下以下这幅图进行理解:

从上图中可以看到:

        1. Throwable:是异常体系的顶层类,其派生出两个重要的子类, Error 和 Exception;
        2. Error:指的是Java虚拟机无法解决的严重问题,比如:JVM的内部错误、资源耗尽等,典型代表:StackOverflowError和OutOfMemoryError,一旦发生回力乏术;
        3. Exception:异常产生后程序员可以通过代码进行处理,使程序继续执行。比如:感冒、发烧。我们平时所说的异常就是Exception;

        

        (2)异常的分类

        在Java中,异常可能在编译时发生,也可能在程序运行时发生,根据发生的时机不同,可以将异常分为两类:

1.编译时异常

        这是因为程序员必须处理这种异常,否则编译器会报错。在Java中最典型的编译时异常为FileNotFoundException,它会在尝试打开不存在的文件时抛出。

import java.io.File;
import java.io.FileReader;
import java.io.FileNotFoundException;

public class CompileTimeExceptionExample {
    public static void main(String[] args) {
        try {
            File file = new File("nonexistent_file.txt"); // 这里假设文件不存在
            FileReader fr = new FileReader(file); // 可能抛出 FileNotFoundException
        } catch (FileNotFoundException e) {
            System.out.println("文件未找到异常:" + e.getMessage());
        }
    }
}

注意:

        编译时出现的语法性错误,不能称之为异常。例如将 System.out.println 拼写错了, 写成了system.out.println. 此时编译过程中就会出错, 这是 "编译期" 出错。而运行时指的是程序已经编译通过得到class 文件了, 再由 JVM 执行过程中出现的错误.

2.运行时异常:

        这些异常不需要强制捕获或处理,因为它们通常是由程序逻辑错误引起的。一个常见的运行时异常是 ArrayIndexOutOfBoundsException,当尝试访问数组中不存在的索引时会抛出。

public class RunTimeExceptionExample {
    public static void main(String[] args) {
        int[] array = {1, 2, 3};
        try {
            System.out.println(array[5]); // 可能抛出 ArrayIndexOutOfBoundsException
        } catch (ArrayIndexOutOfBoundsException e) {
            System.out.println("数组越界异常:" + e.getMessage());
        }
    }
}

这样我们就大致的了解了异常的体系结构了。

3.异常的抛出

        当然,在编写程序时,如果程序中出现错误,此时就需要将错误的信息告知给调用者,比如:参数检测。在Java中,可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。具体语法如下:

throw new XXXException("异常产生的原因");

让我们看一下一个案例:

public static void validateAge(int age) {
        if (age < 18) {
            //异常的抛出
            throw new InvalidAgeException("未满18岁,不允许访问。");
        }
        System.out.println("年龄验证通过。");
}

【注意事项】

1. throw必须写在方法体内部
2. 抛出的对象必须是Exception 或者 Exception 的子类对象
3. 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理
4. 如果抛出的是编译时异常,用户必须处理,否则无法通过编译
5. 异常一旦抛出,其后的代码就不会执行

         这样我们就大致的了解了如果有异常之后,我们如果将其进行抛出了。但是异常是抛出了,那我们如何去解决抛出的异常呢?

4.异常的解决方式

        ——在Java中异常的具体处理方式主要有两种:异常声明 throws 以及 try-catch捕获处理。

        (1)异常声明throws

        当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常。

其语法形式为:

修饰符 返回值类型 方法名(参数列表) throws 异常类型1,异常类型2...{


}

让我们直接使用一个案例来进一步帮助你进行理解:

public static void divide(int a, int b) throws ArithmeticException {
        if (b == 0) {
            throw new ArithmeticException("除数不能为0");
        }
        int result = a / b;
        System.out.println("结果:" + result);
}

        我们可以在这个案例中看到,如果b == 0时,我们抛出了异常,但是我们并没有在此方法中对该异常进行处理,而是将其继续抛出,给调用者处理。这就是第一种处理异常的方throws。

【注意事项】

1. throws必须跟在方法的参数列表之后
2. 声明的异常必须是 Exception 或者 Exception 的子类
3. 方法内部如果抛出了多个异常,throws之后必须跟多个异常类型,之间用逗号隔开,如果抛出多个异常类型具有父子关系,直接声明父类即可。

4. 调用声明抛出异常的方法时,调用者必须对该异常进行处理,或者继续使用throws抛出

        (2)使用try - catch - finally语句

        ——在Java中,异常可以通过try-catch-finally语句块来进行处理。由于throws对异常并没有真正处理,而是将异常报告给抛出异常方法的调用者,由调用者处理。如果真正要对异常进行处理,就需要try-catch。

其基本语法为:

try {
    // 可能抛出异常的代码
} catch (ExceptionType1 e1) {
    // 处理异常类型1的情况
} catch (ExceptionType2 e2) {
    // 处理异常类型2的情况
} finally {
    // 可选的finally语句块,用于执行清理操作
}

        在try语句块中,我们可以放置可能抛出异常的代码。如果try语句块中的代码抛出了异常,那么异常会被catch语句块捕获。catch语句块中的参数指定了要捕获的异常类型,当抛出的异常与catch语句块中指定的类型匹配时,相应的catch语句块会被执行。如果try语句块中的代码未抛出异常,则catch语句块会被跳过。

        另外,finally语句块是可选的,用于执行清理操作,无论是否发生异常都会执行。finally语句块通常用于释放资源或执行必要的收尾工作,例如关闭文件或数据库连接。

这里我们使用一个案例来帮助你进一步理解:

public class SimpleExceptionHandlingExample {

    public static void main(String[] args) {
        int a = 10;
        int b = 0;
        int result = 0;
        try {
            // 尝试进行除法运算
            result = a / b;
            System.out.println("除法运算结果:" + result);
        } catch (ArithmeticException e) {
            // 捕获除零异常并输出错误信息
            System.err.println("除法运算出错:除数不能为零");
        } finally {
            // 无论是否发生异常,都会执行的代码块
            System.out.println("finally块:结束异常处理");
        }
    }
}

        首先我们对10除以0进行除法运算。由于除数为0,会抛出ArithmeticException异常。我们在try块中捕获这个异常,并输出错误信息。无论是否发生异常,finally块中的代码都会执行,这里我们简单输出一条结束异常处理的消息。

【注意事项】

1. try块内抛出异常位置之后的代码将不会被执行
2. 如果抛出异常类型与catch时异常类型不匹配,即异常不会被成功捕获,也就不会被处理,继续往外抛,直到JVM收到后中断程序----异常是按照类型来捕获的

3. try中可能会抛出多个不同的异常对象,则必须用多个catch来捕获----即多种异常,多次捕获

4. 如果多个异常的处理方式是完全相同,我们可以使用“ | ”隔开

5. 如果异常之间具有父子关系,一定是子类异常在前catch,父类异常在后catch,否则语法错误

这里我们对第四条加以解释:如果多个异常的处理方式是完全相同,我们可以使用“ | ”隔开

例如:

catch (ArrayIndexOutOfBoundsException | NullPointerException e) {
    //对异常进行处理的代码
}

        ——当然我们可以一直 | 添加异常,但是实际上不建议使用这样的方式处理,因为这样处理我们将不知道到底try中代码出现了什么类型的异常     (比如上面的代码可能出现的异常类型为ArrayIndexOutOfBoundsException ,也可能是 NullPointerException异常类型

这样我们就大致的了解了Java处理异常的两种方式了。

补充:异常处理流程总结

        1. 程序先执行 try 中的代码.
        2. 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
        3. 如果找到匹配的异常类型, 就会执行 catch 中的代码.
        4. 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
        5. 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
        6. 如果上层调用者也没有处理的了异常, 就继续向上传递.
        7. 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.

5.自定义异常

说到自定义异常,我们首先要了解一下什么是自定义异常:

        自定义异常是指由开发人员根据实际需求创建的异常类,它们扩展自Java标准库中的Exception或其子类。与标准异常类不同,自定义异常类可以提供更具体的异常信息、更详细的上下文信息,以及针对特定业务逻辑的错误处理逻辑。

了解完了什么是自定义异常之后,先让我们看一下如何去创建一个自定义异常。

创建一个自定义异常类,方式如下:

        1. 自定义异常类,然后继承自Exception 或者 RunTimeException
        2. 实现一个带有String类型参数的构造方法,参数含义:出现异常的原因

例子:

// 自定义运行时异常类
class InvalidAgeException extends RuntimeException {
    // 无参构造方法
    public InvalidAgeException() {
        super("年龄无效");
    }

    // 带有异常消息的构造方法
    public InvalidAgeException(String message) {
        super(message);
    }
}

        这样我们就创建好了一个自定义异常了,那么自定义异常创建好之后,我们又该如何的使用它呢?让我们继续往下看!

现在我们将要使用刚才创建好的自定义异常类来完成一个小案例:(代码如下)

// 创建人类
class Person {
    private String name;
    private int age;

    // 构造方法
    public Person(String name, int age) {
        this.name = name;
        // 检查年龄是否有效,如果无效则抛出自定义异常
        if (age < 0 || age > 150) {
            //使用了我们刚才创建的自定义异常类
            throw new InvalidAgeException("年龄必须在0到150之间");
        }
        this.age = age;
    }

    // 获取年龄
    public int getAge() {
        return age;
    }
}

//测试类
public class CustomRuntimeExceptionExample {
    public static void main(String[] args) {
        // 创建一个人类对象,年龄为负数,会抛出自定义异常
        try {
            Person person = new Person("Alice", -10);
            System.out.println(person.getAge());
        } catch (InvalidAgeException e) {
            // 捕获并处理自定义异常
            System.err.println("创建人物失败:" + e.getMessage());
        }

        // 创建一个人类对象,年龄超过有效范围,也会抛出自定义异常
        try {
            Person person = new Person("Bob", 160);
            System.out.println(person.getAge());
        } catch (InvalidAgeException e) {
            // 捕获并处理自定义异常
            System.err.println("创建人物失败:" + e.getMessage());
        }
    }
}

        在上面的代码中,我们首先定义了一个名为InvalidAgeException的自定义异常类,它继承自RuntimeException类。然后我们定义了一个Person类,其中包含一个构造方法用于创建人类对象,并在构造方法中检查年龄是否有效。如果年龄无效,则抛出InvalidAgeException异常。

        然后在main方法中,我们分别创建了两个Person对象,一个年龄为负数,一个年龄超过有效范围。在创建这两个对象时,都会抛出自定义异常。我们使用try-catch语句块捕获并处理这些异常,输出错误信息到控制台。

        通过这个案例我相信读者你已经大致的了解了如何创建一个自定义异常类和如何去使用创建的自定义异常类了。

注意事项:

1. 自定义异常通常会继承自 Exception 或者 RuntimeException
2. 继承自 Exception 的异常默认是受查异常
3. 继承自 RuntimeException 的异常默认是非受查异常.

这样我们就了解了Java中自定义异常了。


以上就是本篇文章的所以内容了~~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/674722.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

什么文档加密软件好用?迅软DSE加密软件你不会还不知道吧?

一、什么文档加密软件好用&#xff1f; 其中有迅软DSE文档加密软件等。 迅软DSE加密软件&#xff1a;让企业的创意成果、招投标文件、生产工艺、流程配方、研发成果、公司计划、员工信息等核心数据更安全。 多方面加密模式&#xff0c;有效防止数据泄密 透明无感知加密&…

数据结构C语言版 严蔚敏

下列关于数据的逻辑结构的叙述中&#xff0c;哪一个是不正确的&#xff08;C&#xff09;。 A、数据的逻辑结构是数据间关系的描述B、数据的逻辑结构抽象反映数据元素间的逻辑关系C、数据的逻辑结构具体反映数据在计算机中的存储方式 数据的逻辑结构是从逻辑关系上描述数据&am…

华为云Pass平台微服务治理

华为云Pass平台微服务治理 1. 快速体验 1.1 微服务结构 2. 搭建项目 远程调用 主启动类 RequestSchema指定controller注解并标注控制类名称 3. PASS平台 CSE ServiceComb接入华为云CSE 注意&#xff1a; 当你的接口出现变更&#xff0c;把微服务引擎服务目录重新删除&…

深入理解mysql中的各种超时属性

1. 前言 connectTimeout: 连接超时 loginTimeout: 登录超时 socketTimeout: Socket网络超时&#xff0c;即读超时 queryTimeout: sql执行超时 transactionTimeout:spring事务超时 innodb_lock_wait_timeout:innodb锁等待超时 netTimeoutForStreamingResults:mysql server网络回…

U盘感染病毒,不必急于扔掉!教你如何有效清除U盘中的病毒

U盘被感染了只能扔掉吗&#xff1f;随着信息时代的飞速发展&#xff0c;U盘已成为我们日常生活中不可或缺的存储设备。然而&#xff0c;在使用U盘的过程中&#xff0c;我们有时会遇到U盘被病毒感染的情况。面对这一问题&#xff0c;许多人可能会感到恐慌&#xff0c;甚至想要直…

Ubuntu22.04之解决:terminal使用alt+1/alt+2/alt+3失效问题(二百三十八)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

UML实现图-组件图

概述 组件图(ComponentDiagram)描述了软件的各种组件和它们之间的依赖关系。组件图中通常包含4种元素:组件、程序、包、任务&#xff0c;各个组件之间还可以相互依赖。 一、组件的表示法 组件是定义了良好接口的物理实现单元&#xff0c;是系统中可替换的物理部件。在一般情…

动物群体甲基化如何讲故事?

DNA甲基化是DNA化学修饰的一种形式&#xff0c;能够在不改变DNA序列的前提下&#xff0c;改变基因表达&#xff0c;从而使表型发生变化。DNA甲基化是一种非常保守的表观遗传修饰&#xff0c;其广泛存在于生物体内&#xff0c;在不同的物种间、相同物种不同环境&#xff0c;同一…

学完PMP,怎么应用到实际工作中?

学习理论知识只是第一步&#xff0c;其目的是为了在实际工作中能够提供帮助。我相信题主的问题也是许多人共同面临的。PMP的知识体系是专业项目管理所必须掌握的知识体系。俗话说&#xff0c;有备无患&#xff0c;具备专业的理论知识就是最好的准备。废话不多说&#xff0c;举几…

使用 zxing 生成二维码以及条形码

需求背景 前期在做项目的时候&#xff0c;有一个需求是说要生成一张条形码&#xff0c;并且呢将条形码插入到 excel 中去&#xff0c;但是之前一直没有搞过找个条形码或者是二维码&#xff0c;最后是做出来了&#xff0c;这里呢就先看看怎么生成&#xff0c;后面再抽时间来写写…

6.3 cf 944

Problem - C - Codeforces 思路 分四种情况&#xff0c;以12为分界点 &#xff08;紫色部分是最初思路&#xff0c;但不包含所有情况&#xff09; 只看在a<b c<d 时的图 代码 #include <bits/stdc.h> using namespace std; #define IOS ios::sync_with_stdio(…

web前端三大主流框架指的是什么

web前端三大主流框架是什么&#xff1f;前端开发师的岗位职责有哪些&#xff1f;这边整理了相关内容供大家参考了解&#xff0c;请各位小伙伴随小编一起查阅下面的内容。 web前端三大主流框架 web前端三大主流框架是Angular、React、Vue。 1.Angular Angular原名angularJS诞生…

OJ3376无尽的石头问题

答案&#xff1a; #include<bits/stdc.h> using namespace std; const int N10e7; int fx(int n) {int sum0;while(n){sum(n%10);n/10;}return sum; } int main() {int t,n,x;cin>>t;while(t--){cin>>n;int count0;for(int i1;i<N;){if(in){cout<<…

网关(Gateway)- 自定义过滤器工厂

自定义过滤工厂类 DemoGatewayFilterFactory package com.learning.springcloud.custom;import org.apache.commons.lang.StringUtils; import org.springframework.cloud.gateway.filter.GatewayFilter; import org.springframework.cloud.gateway.filter.GatewayFilterChai…

FreeRTOS【16】直达任务通知使用

1.开发背景 直达任务通知&#xff0c;FreeRTOS 的线程任务提供的接口&#xff0c;可以用作线程唤醒&#xff0c;或者是传递数据&#xff0c;因为是基于线程本身的操作&#xff0c;是轻量级&#xff0c;速度响应更快&#xff0c;适合小内存芯片使用。 事实上本人使用得比较少&am…

pytorch笔记:自动混合精度(AMP)

1 理论部分 1.1 FP16 VS FP32 FP32具有八个指数位和23个小数位&#xff0c;而FP16具有五个指数位和十个小数位Tensor内核支持混合精度数学&#xff0c;即输入为半精度&#xff08;FP16&#xff09;&#xff0c;输出为全精度&#xff08;FP32&#xff09; 1.1.1 使用FP16的优缺…

大渡口数字经济产业商会暨尼伽OLED透明屏产品发布会

2024年5月31日&#xff0c;大渡口数字经济产业商会成功举办了一场盛大的“商会数字经济发展项目签约大会”&#xff0c;活动上不仅深入探讨了构建“义渡新质生产力”及如何更好地“建功重庆西部大开发”的战略议题&#xff0c;还正式与尼伽OLED宣布达成战略合作伙伴关系&#x…

Java版工程项目管理系统源码:技术框架与功能实现全解析

在工程行业&#xff0c;项目管理的高效协同和信息共享是提升管理效率和精度的关键。本文将详细介绍一款采用先进技术框架的Java版工程项目管理系统&#xff0c;该系统支持前后端分离&#xff0c;功能全面&#xff0c;可满足不同角色的需求。从项目进度图表到施工地图&#xff0…

10个从基础到高级的GPT提示词优化指南

为一名大模型的深度用户和微软Copilot的首批开放测试者&#xff0c;很多人会问我如何写出高效的提示词。同时&#xff0c;也有不少读者反映&#xff0c;像ChatGPT和Claude这样的模型并没有想象中那么神奇&#xff0c;无法满足他们的实际需求。 首先我想说&#xff0c;确实像Ch…

双指针_复写零

复写零 题目描述&#xff1a; 题目链接&#xff1a;复写零 内容&#xff1a; 这道题目要求我们每遇到一次0就复写一遍&#xff0c;并且只能在原数组上进行修改&#xff0c;不能越界访问。 算法原理&#xff1a; 思路1&#xff1a; 如果我们用两个指针cur,dest同时从指向第一个…