Java 异常处理全解析:从基础到自定义异常的实战指南

Java 异常处理全解析:从基础到自定义异常的实战指南

一、Java 异常体系:Error 与 Exception 的本质区别

1. 异常体系核心架构

Java把异常当作对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类。

在Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception

Java 异常体系基于Throwable类,分为两大分支:

Throwable
├─ Error(错误)
│  └─ 例:OutOfMemoryErrorStackOverflowError
└─ Exception(异常)├─ Checked Exception(编译时异常)│  └─ 例:IOExceptionSQLException└─ Unchecked Exception(运行时异常)└─ 例:NullPointerExceptionArrayIndexOutOfBoundsException

在这里插入图片描述

2.什么是异常以及异常的简单分类

  • 异常指程序运行中出现的不期而至的各种状况,如:文件找不到、网络连接失败、非法参数等。
  • 异常发生在程序运行期间,它影响了正常的程序执行流程。

要理解java异常处理是如何工作的,你需要掌握以下三种类型的异常Exception

  1. 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
  2. 运行时异常:运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
  3. 错误Error: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。

3. Error vs Exception

Error和Exception的区别:Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,Java虚拟机(JVM)一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。

维度ErrorException
本质JVM 底层错误,不可恢复程序运行中的异常情况
处理方式无法捕获,需重启 JVM可捕获处理或声明抛出
典型场景内存溢出、类加载失败空指针、数组越界、用户输入错误

4. Checked vs Unchecked 异常

  • Checked 异常

    • 必须显式处理(捕获或声明抛出)
    • 示例:文件读取时的FileNotFoundException
    try {FileReader reader = new FileReader("file.txt");
    } catch (FileNotFoundException e) {e.printStackTrace();
    }
    
  • Unchecked 异常

    • 无需强制处理(运行时检查)
    • 示例:参数校验失败的IllegalArgumentException
    public void checkAge(int age) {if (age < 0) {throw new IllegalArgumentException("年龄不能为负数");}
    }
    

二、异常处理核心机制:捕获与抛出

1. 异常捕获语法(try-catch-finally)

try {// 可能抛出异常的代码int result = 10 / 0; // 抛出ArithmeticException
} catch (ArithmeticException e) { // 捕获具体异常System.out.println("错误原因:" + e.getMessage()); // 输出:/ by zeroe.printStackTrace(); // 打印堆栈跟踪信息
} catch (Exception e) { // 父类异常捕获(需放在最后)// 通用异常处理
} finally {// 可选:无论是否异常都会执行(常用于资源释放)System.out.println("执行finally块");
}

2. 异常抛出关键字

throw(方法内抛出异常实例)
public static int divide(int a, int b) {if (b == 0) {throw new ArithmeticException("除数不能为0"); // 主动抛出异常}return a / b;
}
throws(方法声明抛出异常)
public static void readFile(String path) throws FileNotFoundException {FileReader reader = new FileReader(path); // 声明抛出Checked异常
}

3. 异常处理顺序原则

  • 子类异常优先:具体异常(如FileNotFoundException)需写在父类异常(如IOException)之前
  • finally 的绝对性:即使 try 块中有return,finally 仍会执行(但返回值会被缓存)

4.本节狂神说笔记

package com.exception;public class Test {public static void main(String[] args) {int a = 10;int b = 0;//假设要捕获多个异常: 从小到大!try {//try监控区域System.out.println(a/b);} catch (Error e) {//catch(想要捕获的异常类型!)捕获异常System.out.println("Error");} catch (Exception e) {System.out.println("Exception");} catch (Throwable e) {System.out.println("Throwable");} finally {//处理善后工作System.out.println("finally");}finally 可以不要finally, 假设I0,资源,关闭!}public void a(){b();}public void b(){a();}
}
package com.exception;public class Test2 {public static void main(String[] args) {int a = 1;int b = 0;//Ctrl Alt + Ttry {System.out.println(a/b);} catch (Exception e) {e.printStackTrace();//打印错误的栈信息} finally {}}
}
package com.exception;public class Test1 {public static void main(String[] args) {try {new Test1().test(1,0);} catch (ArithmeticException e) {e.printStackTrace();}}//假设这个方法中,处理不了这个异常。方法上抛出异常public void test(int a,int b) throws ArithmeticException{if(b == 0){//throw  throwsthrow new ArithmeticException();//主动的抛出异常,一般在方法中使用}}
}

三、自定义异常:打造业务专属错误体系

使用Java内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需继承Exception类即可。

1. 自定义异常类步骤

  1. 创建自定义异常类。
  2. 在方法中通过throw关键字抛出异常对象。
  3. 如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
  4. 在出现异常方法的调用者中捕获并处理异常。
步骤 1:继承 Exception 或 RuntimeException
// Checked异常(需显式处理)
public class UserAgeException extends Exception {public UserAgeException(String message) {super(message); // 调用父类构造器}
}// Unchecked异常(运行时异常)
public class InvalidInputException extends RuntimeException {public InvalidInputException(String message) {super(message);}
}
步骤 2:在业务逻辑中抛出
public void validateUser(int age) throws UserAgeException {if (age < 18) {throw new UserAgeException("用户年龄必须≥18岁"); // 抛出自定义Checked异常}
}

2. 异常处理最佳实践

(1)异常信息规范
  • 包含足够上下文:"用户ID为" + userId + "的账户不存在"
  • 避免裸露捕获Exception:应捕获具体异常类型
(2)资源释放最佳实践
// JDK 7+ 自动资源管理(替代finally)
try (FileReader reader = new FileReader("file.txt")) {// 自动关闭资源(无需显式finally)
} catch (IOException e) {// 处理文件读取异常
}
(3)异常链使用
try {// 业务逻辑
} catch (ServiceException e) {throw new ControllerException("接口调用失败", e); // 包装异常链
}

3.实际应用中的经验总结

  • 处理运行时异常时,采用逻辑去合理规避同时辅助 try-catch 处理
  • 在多重catch块后面,可以加一个catch(Exception)来处理可能会被遗漏的异常
  • 对于不确定的代码,也可以加上 try-catch,处理潜在的异常
  • 尽量去处理异常,切忌只是简单地调用 printStackTrace()去打印输出
  • 具体如何处理异常,要根据不同的业务需求和异常类型去决定
  • 尽量添加finally语句块去释放占用的资源

四、常见异常处理错误与解决方案

1. 空指针异常(NPE)

错误示例

String str = null;
int length = str.length(); // 抛出NullPointerException

解决方案

if (str != null) {int length = str.length();
} else {throw new IllegalArgumentException("字符串不可为null");
}

2. 未处理 Checked 异常

错误示例

public void readFile() {FileReader reader = new FileReader("file.txt"); // 编译错误:未处理IOException
}

解决方案

// 方案1:捕获处理
try { /* ... */ } catch (IOException e) { /* ... */ }// 方案2:声明抛出
public void readFile() throws IOException { /* ... */ }

3. finally 中的 return 陷阱

错误示例

public static int testFinally() {try {return 1;} finally {return 2; // 最终返回2,覆盖try中的return}
}

正确做法

public static int testFinally() {int result = 1;try {return result;} finally {result = 2; // 不建议在finally中使用return}
}

五、高频面试题解析

1. Error 和 Exception 的根本区别?

  • Error 是 JVM 内部错误(如内存溢出),无法通过代码处理,必须重启应用
  • Exception 是程序运行中的异常,分为 Checked(编译时检查)和 Unchecked(运行时异常)

2. throw 和 throws 的区别?

关键字作用使用位置参数类型
throw抛出异常实例方法体内部异常对象
throws声明方法可能抛出的异常类型方法签名异常类列表

3. finally 块一定会执行吗?

  • 正常情况下一定会执行(包括 return 前执行)
  • 唯一例外:JVM 提前终止(如System.exit(0)

4. 自定义异常应该继承哪个类?

  • 业务需要编译时检查:继承Exception
  • 运行时异常(如参数校验失败):继承RuntimeException

六、异常处理核心原则

  1. 具体捕获:优先捕获具体异常,避免使用宽泛的Exception
  2. 快速失败:在方法入口校验参数,尽早抛出异常
  3. 信息完整:异常信息需包含定位问题的关键数据(如用户 ID、时间戳)
  4. 资源管理:使用try-with-resources自动释放 IO、数据库连接等资源

七、总结:构建健壮的异常处理体系

通过合理运用 Java 异常处理机制,开发者可以:

  1. 清晰区分系统错误与业务异常
  2. 通过捕获特定异常实现精准的错误处理
  3. 利用自定义异常提升业务代码的可读性
    该继承哪个类?
  • 业务需要编译时检查:继承Exception
  • 运行时异常(如参数校验失败):继承RuntimeException

六、异常处理核心原则

  1. 具体捕获:优先捕获具体异常,避免使用宽泛的Exception
  2. 快速失败:在方法入口校验参数,尽早抛出异常
  3. 信息完整:异常信息需包含定位问题的关键数据(如用户 ID、时间戳)
  4. 资源管理:使用try-with-resources自动释放 IO、数据库连接等资源

七、总结:构建健壮的异常处理体系

通过合理运用 Java 异常处理机制,开发者可以:

  1. 清晰区分系统错误与业务异常
  2. 通过捕获特定异常实现精准的错误处理
  3. 利用自定义异常提升业务代码的可读性
  4. 结合 finally 和 try-with-resources 确保资源安全释放

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

redis 数据类型新手练习系列——string类型

redis 数据类型 Redis 主要支持以下几种数据类型&#xff1a; &#xff08;1&#xff09;string&#xff08;字符串&#xff09;: 基本的数据存储单元&#xff0c;可以存储字符串、整数或者浮点数。 &#xff08;2&#xff09;hash&#xff08;哈希&#xff09;:一个键值对集…

Android12源码编译及刷机

由于google的AOSP源码拉取经常失败&#xff0c;编译还经常出现各种问题。这里根据香橙派Orange Pi 5 Plus&#xff08;Android12电视镜像&#xff09;源码进行编译演示。 RK芯片的开发板可玩性很高&#xff0c;这里以电视版本android系统为例子&#xff0c;学习的同时还可以当…

从零实现 registry.k8s.io/pause:3.8 镜像的导出与导入

以下是为 registry.k8s.io/pause:3.8 镜像的导出与导入操作定制的完整教程&#xff0c;适用于 Kubernetes 集群中使用 containerd 作为容器运行时的场景。本教程包含详细步骤、常见问题解析及注意事项。 从零实现 registry.k8s.io/pause:3.8 镜像的导出与导入 背景说明 Kuber…

Redis和MQ的区别

redis是一个高性能的key-value数据库&#xff0c;支持消息推送功能&#xff0c;可以当做一个轻量级的队列服务器使用。 redis只是提供一个高性能的、原子操作内存键值队&#xff0c;具有高速访问能力&#xff0c;虽然可以做消息队列的存储&#xff0c;但不具备消息队列的任何功…

Centos7系统防火墙使用教程

CentOS 7是一种常见的Linux操作系统&#xff0c;防火墙作为网络安全的第一道防线&#xff0c;对于服务器的安全至关重要。本文将介绍CentOS 7系统中防火墙的使用教程&#xff0c;包括如何开启、关闭、配置以及防火墙规则的添加和删除。 一、查看防火墙状态 在开始操作之前&am…

Uniapp:navigator(页面跳转)

目录 一、基本概述二、属性说明三、具体使用一、基本概述 页面跳转。该组件类似HTML中的<a>组件,但只能跳转本地页面。目标页面必须在pages.json中注册。 二、属性说明 属性名类型默认值说明平台差异说明urlString应用内的跳转链接,值为相对路径或绝对路径,如:“……

大疆机场及无人机上云(航线规划、指令飞行...)

系统操作预览&#xff1a; 包含一键起飞、指令飞行、云台控制、变焦、航线规划、空域规划、成果数据展示、实时飞行模拟、任务派发等 大疆无人机飞控平台&#xff08;航线规划、机场3、私有化部署&#xff09;_哔哩哔哩_bilibili 2025-04-02 更新 start、 已支持大疆机场3。…

【运维】云端掌控:用Python和Boto3实现AWS资源自动化管理

《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 在云计算时代,AWS(Amazon Web Services)作为领先的云服务平台,其资源管理的高效性对企业至关重要。本文深入探讨如何利用Python的boto3…

Java ThreadLocal与内存泄漏

当我们利用 ThreadLocal 来管理数据时&#xff0c;我们不可避免地会面临内存泄漏的风险。 原因在于 ThreadLocal 的工作方式。当我们在当前线程的 ThreadLocalMap 中存储一个值时&#xff0c;一旦这个值不再需要&#xff0c;释放它就变得至关重要。如果不这样做&#xff0c;那么…

Linux之netlink(2)libnl使用介绍(1)

Linux之netlink(2)Libnl3使用介绍(1) Author&#xff1a;Onceday Date&#xff1a;2025年4月26日 漫漫长路&#xff0c;才刚刚开始… 全系列文章可查看专栏: Linux内核知识_Once-Day的博客-CSDN博客 本文翻译自libnl3官方文档&#xff1a;Netlink Library (libnl) 参考文档…

深入理解TransmittableThreadLocal:原理、使用与避坑指南

一、ThreadLocal与InheritableThreadLocal回顾 在介绍TransmittableThreadLocal之前&#xff0c;我们先回顾一下Java中的ThreadLocal和InheritableThreadLocal。 1. ThreadLocal ThreadLocal提供了线程局部变量&#xff0c;每个线程都可以通过get/set访问自己独立的变量副本…

Linux下的I/O复用技术之epoll

I/O多路复用 指在单个线程或进程中&#xff0c;同时处理多个I/O操作的技术。 旨在提高程序处理多个并发I/O操作的能力&#xff0c;避免程序因等待某个I/O操作而被阻塞。在传统的I/O模型中当程序进行I/O操作时(如读取文件、接受网路数据等)&#xff0c;如果数据还未准备好&…

用 C 语言实现通用的冒泡排序算法

在日常编程中&#xff0c;排序算法是一个非常常见且重要的工具。虽然有许多排序算法可以选择&#xff0c;但如果你需要一个能够处理不同数据类型的排序算法&#xff0c;如何设计一个通用的排序算法呢&#xff1f;今天我们将实现一个通用的冒泡排序算法&#xff0c;支持不同数据…

C# 变量全解析:声明、初始化与使用

在多用途的编程语言中&#xff0c;程序存取数据是一项基础且关键的功能&#xff0c;而这一功能主要通过变量来实现。本文将全面深入地探讨 C# 中的变量&#xff0c;包括变量的种类、声明、初始化、自动初始化、多变量声明以及如何使用变量的值。 变量概述 变量是一个名称&…

Dify中的文本分词处理技术详解

Dify中的文本分词处理技术详解 引言核心架构概览索引处理器工厂 文本分词技术详解基础分词器增强型递归字符分词器固定分隔符文本分词器递归分割算法 索引处理器中的分词应用特殊索引处理器的分词特点问答索引处理器父子索引处理器 分词技术的应用场景技术亮点与优势总结 引言 …

如何打包python程序为可执行文件

将 Python 程序打包为可执行文件是一个常见需求&#xff0c;尤其是在希望将应用程序分享给不具备 Python 环境的用户时。以下是使用 PyInstaller 工具将 Python 程序打包为可执行文件的步骤。 步骤 1&#xff1a;安装 PyInstaller 如果您还没有安装 PyInstaller&#xff0c;请…

美团Java后端二面面经!

场景题是面试的大头&#xff0c;建议好好准备 Q. [美团]如何设计一个外卖订单的并发扣减库存系统&#xff1f; Q.[美团]为啥初始标记和重新标记需要STW&#xff1f; Q.[美团]骑手位置实时更新&#xff0c;如何保证高并发写入&#xff1f; Q.[美团]订单表数据量过大导致查询…

在应用运维过程中,业务数据修改的证据留存和数据留存

在应用运维过程中,业务数据修改的证据留存和数据留存至关重要,以下是相关介绍: 一、证据留存 操作日志记录 : 详细记录每一次业务数据修改的操作日志,包括操作人员、操作时间、修改内容、修改前后数据的对比等。例如,某公司业务系统中,操作日志会精确记录员工小张在 2…

Eigen迭代求解器类

1. 迭代求解器核心类概览 Eigen 提供多种迭代法求解稀疏线性方程组 AxbAxb&#xff0c;适用于大规模稀疏矩阵&#xff1a; 求解器类适用矩阵类型算法关键特性ConjugateGradient对称正定&#xff08;SPD&#xff09;共轭梯度法&#xff08;CG&#xff09;高精度&#xff0c;内…