在前六天的学习中,我们掌握了 Java 的基础语法、面向对象核心特性、抽象类与接口等知识。今天我们将学习 Java 中的异常处理机制,这是保证程序健壮性的关键技术。在 JavaWeb 开发中,无论是用户输入错误、数据库连接失败还是网络异常,都需要通过异常处理机制来优雅地处理,避免程序崩溃并给用户友好提示。

什么是异常?

异常(Exception)是程序运行过程中发生的意外情况,它会中断程序的正常执行流程。例如:

  • 除以零(ArithmeticException
  • 访问数组越界(ArrayIndexOutOfBoundsException
  • 读取不存在的文件(FileNotFoundException
  • 网络连接中断

没有异常处理的程序遇到异常时会直接崩溃,例如:

public class TestWithoutException {public static void main(String[] args) {int a = 10;int b = 0;System.out.println(a / b); // 发生算术异常System.out.println("程序继续执行"); // 这行代码不会执行}
}

运行结果:

Exception in thread "main" java.lang.ArithmeticException: / by zeroat TestWithoutException.main(TestWithoutException.java:5)

异常的分类

Java 中的异常体系以Throwable为根类,主要分为两大类:

  1. Error(错误)

    • 由 JVM 产生的严重错误,程序无法处理(如内存溢出OutOfMemoryError
    • 通常不需要捕获,需要从代码层面解决
  2. Exception(异常)

    • 程序可以处理的异常,分为:
      • 编译时异常(受检异常):编译期间必须处理的异常(如IOExceptionSQLException
      • 运行时异常(非受检异常):运行时才会发生的异常(如NullPointerExceptionIndexOutOfBoundsException

异常体系结构简图:

Throwable
├─ Error(错误)
│  ├─ OutOfMemoryError
│  └─ StackOverflowError
│
└─ Exception(异常)├─ RuntimeException(运行时异常)│  ├─ NullPointerException│  ├─ ArithmeticException│  └─ IndexOutOfBoundsException│├─ IOException(编译时异常)├─ SQLException(编译时异常)└─ ClassNotFoundException(编译时异常)

异常处理的核心语法

Java 提供了try-catch-finallythrows关键字来处理异常,确保程序在遇到异常时能够继续执行或优雅退出。

1. try-catch-finally 语句

try {// 可能发生异常的代码块
} catch (异常类型1 变量名1) {// 处理异常类型1的代码
} catch (异常类型2 变量名2) {// 处理异常类型2的代码
} finally {// 无论是否发生异常,都会执行的代码(通常用于资源释放)
}

执行流程

  • 如果try块中没有异常,执行try块后直接执行finally
  • 如果try块中有异常,中断try块执行,匹配对应的catch块处理,最后执行finally

实例:

public class TryCatchDemo {public static void main(String[] args) {int[] nums = {1, 2, 3};try {// 可能发生异常的操作int result = 10 / 0; // 算术异常System.out.println(nums[3]); // 数组越界异常(不会执行)} catch (ArithmeticException e) {// 处理算术异常System.out.println("捕获到算术异常:" + e.getMessage());e.printStackTrace(); // 打印异常堆栈信息(调试用)} catch (ArrayIndexOutOfBoundsException e) {// 处理数组越界异常System.out.println("捕获到数组越界异常:" + e.getMessage());} finally {// 无论是否异常,都会执行System.out.println("finally块执行:资源释放操作");}// 异常处理后,程序可以继续执行System.out.println("程序继续运行...");}
}

运行结果:

捕获到算术异常:/ by zero
java.lang.ArithmeticException: / by zeroat TryCatchDemo.main(TryCatchDemo.java:8)
finally块执行:资源释放操作
程序继续运行...

2. throws 声明异常

当方法内部无法处理异常时,可以使用throws关键字声明该方法可能抛出的异常,由调用者处理:

// 声明方法可能抛出的异常
修饰符 返回值类型 方法名(参数列表) throws 异常类型1, 异常类型2 {// 可能抛出异常的代码
}

实例:

import java.io.FileInputStream;
import java.io.FileNotFoundException;public class ThrowsDemo {// 声明可能抛出编译时异常public static void readFile(String fileName) throws FileNotFoundException {// 读取文件(可能抛出FileNotFoundException)new FileInputStream(fileName);}public static void main(String[] args) {try {// 调用声明异常的方法,必须处理异常readFile("test.txt");} catch (FileNotFoundException e) {System.out.println("处理文件不存在异常:" + e.getMessage());}}
}

注意

  • 运行时异常可以不声明,但编译时异常必须声明或捕获
  • 子类重写父类方法时,抛出的异常不能超过父类方法声明的异常范围

3. throw 主动抛出异常

使用throw关键字可以在代码中主动抛出异常:

public class ThrowDemo {// 验证年龄的方法public static void checkAge(int age) {if (age < 0 || age > 120) {// 主动抛出异常throw new IllegalArgumentException("年龄不合法:" + age);}System.out.println("年龄验证通过:" + age);}public static void main(String[] args) {try {checkAge(150); // 调用方法} catch (IllegalArgumentException e) {System.out.println("捕获到异常:" + e.getMessage());}}
}

运行结果:

捕获到异常:年龄不合法:150

自定义异常

在实际开发中,系统提供的异常可能无法满足业务需求,这时可以自定义异常类:

  1. 继承Exception(编译时异常)或RuntimeException(运行时异常)
  2. 提供构造方法(通常需要带消息的构造方法)

实例:

// 自定义编译时异常
public class InsufficientFundsException extends Exception {// 无参构造public InsufficientFundsException() {super();}// 带消息的构造public InsufficientFundsException(String message) {super(message);}
}// 自定义运行时异常
public class InvalidAccountException extends RuntimeException {public InvalidAccountException(String message) {super(message);}
}// 使用自定义异常
public class BankService {private double balance; // 账户余额public BankService(double balance) {this.balance = balance;}// 取款方法(可能抛出自定义异常)public void withdraw(double amount) throws InsufficientFundsException {if (amount <= 0) {throw new InvalidAccountException("取款金额不能为负数");}if (amount > balance) {// 抛出编译时异常,必须声明throw new InsufficientFundsException("余额不足,当前余额:" + balance + ",取款:" + amount);}balance -= amount;System.out.println("取款成功,剩余余额:" + balance);}public static void main(String[] args) {BankService bank = new BankService(1000);try {bank.withdraw(1500);} catch (InsufficientFundsException e) {System.out.println("取款失败:" + e.getMessage());} catch (InvalidAccountException e) {System.out.println("操作失败:" + e.getMessage());}}
}

运行结果:

取款失败:余额不足,当前余额:1000.0,取款:1500.0

异常处理的最佳实践

  1. 避免捕获所有异常:不要使用catch (Exception e)捕获所有异常,应该针对性处理

    // 不推荐
    try {// ...
    } catch (Exception e) {// 无法区分具体异常类型
    }
    
  2. 不要忽略异常:捕获异常后必须处理(至少记录日志),避免空的catch

    // 不推荐
    try {// ...
    } catch (IOException e) {// 空块会隐藏错误
    }
    
  3. 释放资源:使用finally块释放资源(如文件流、数据库连接)

    FileInputStream fis = null;
    try {fis = new FileInputStream("test.txt");// 读取文件
    } catch (FileNotFoundException e) {e.printStackTrace();
    } finally {// 确保关闭流if (fis != null) {try {fis.close();} catch (IOException e) {e.printStackTrace();}}
    }
    
  4. Java 7+:try-with-resources:自动释放实现AutoCloseable接口的资源

    // 推荐:自动关闭资源,无需手动调用close()
    try (FileInputStream fis = new FileInputStream("test.txt")) {// 读取文件
    } catch (IOException e) {e.printStackTrace();
    }
    
  5. 使用恰当的异常类型:根据业务场景选择或创建合适的异常类型

异常处理在 JavaWeb 中的应用

在 JavaWeb 开发中,异常处理尤为重要,常见场景包括:

  1. Servlet 中的异常处理

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {try {String username = request.getParameter("username");if (username == null || username.isEmpty()) {throw new IllegalArgumentException("用户名不能为空");}// 处理业务逻辑} catch (IllegalArgumentException e) {// 向客户端返回错误信息response.getWriter().write("错误:" + e.getMessage());response.setStatus(400); // 设置HTTP错误状态码}
    }
    
  2. 全局异常处理器:在 Spring 等框架中,可以定义全局异常处理器统一处理异常,避免代码冗余

总结与实践

知识点回顾

  1. 异常概念:程序运行时的意外情况,会中断正常执行流程
  2. 异常分类Error(无法处理)和Exception(可处理),Exception分为编译时异常和运行时异常
  3. 处理方式
    • try-catch-finally:捕获并处理异常,释放资源
    • throws:声明方法可能抛出的异常,由调用者处理
    • throw:主动抛出异常
  4. 自定义异常:继承ExceptionRuntimeException,满足业务需求
  5. 最佳实践:针对性处理异常、不忽略异常、及时释放资源

实践任务

  1. 用户注册异常处理

    • 创建自定义异常UserAlreadyExistsException(用户已存在)和InvalidUserInfoException(用户信息无效)
    • 编写UserService类,包含register(String username, String password)方法:
      • 如果用户名已存在(可简单判断是否为 "admin"),抛出UserAlreadyExistsException
      • 如果密码长度小于 6 位,抛出InvalidUserInfoException
      • 否则提示注册成功
    • 编写测试类,使用try-catch处理异常并输出相应信息
  2. 文件读取异常处理

    • 编写FileUtil类,包含readFileContent(String filePath)方法,读取文件内容
    • 处理可能的异常(文件不存在、IO 异常等)
    • 使用try-with-resources确保文件流正确关闭

思考:在 Web 开发中,为什么不建议直接将异常堆栈信息返回给客户端?应该如何处理?

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

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

相关文章

编译器默认生成的c++类六大成员函数

编译器默认生成的c类六大成员函数 编译器默认生成的六大成员函数 当你定义一个空类时&#xff0c;例如&#xff1a; class Empty {};如果代码中没有显式定义任何成员函数&#xff0c;C编译器会在需要时&#xff08;例如&#xff0c;代码中实际调用了这些函数&#xff09;为你…

人工智能概念:常见的大模型微调方法

文章目录一、微调技术的底层逻辑1.1 预训练与微调的关系1.2 核心目标&#xff1a;适配任务与数据二、经典微调方法详解2.1 全量微调&#xff08;Full Fine-Tuning&#xff09;2.2 冻结层微调&#xff08;Layer-Freezing Fine-Tuning&#xff09;2.3 参数高效微调&#xff08;Pa…

动态路由协议(一)

1. 动态路由 概述 静态路由在大网络里太麻烦&#xff08;设备多、配置量大&#xff0c;拓扑变了还要手动改&#xff09; 静态路由是由工程师手动配置和维护的路由条目&#xff0c;命令行简单明确&#xff0c;适用于小型或稳定的网络。静态路由有以下问题&#xff1a; 无法适…

LINUX812 shell脚本:if else,for 判断素数,创建用户

问题 [rootweb ~]# for((i2;i<n;i)) > if [ $n -ne $i ] && [ $((n%i)) -eq 0 ];then -bash: 未预期的符号 if 附近有语法错误 您在 /var/spool/mail/root 中有邮件 [rootweb ~]#[rootweb ~]# cat judgeprimeok.sh declare -i n read -p "please type the n…

游戏中角色持枪:玩家操控角色,角色转向时枪也要转向

角色持有枪&#xff0c;玩家&#xff08;你&#xff09;操控角色&#xff0c;那么&#xff0c;在角色转向时&#xff0c;枪也要转向。 先看看简单情况&#xff1a;假定角色只面向左或右方向&#xff0c;pygame中用这句来实现&#xff1a;pos self.facing * self.gun_offset s…

深度学习入门Day8:生成模型革命——从GAN到扩散模型

一、开篇&#xff1a;创造力的算法革命从昨天的Transformer到今天的生成模型&#xff0c;我们正从"理解"世界迈向"创造"世界。生成对抗网络(GAN)和扩散模型(Diffusion Model)代表了当前生成式AI的两大主流范式&#xff0c;它们让机器能够生成逼真的图像、音…

基于WRF-Chem的不同气溶胶的辐射效应的研究

前言目前我对于气溶胶辐射效应的理解就是设计敏感性实验&#xff0c;基础实验打开气溶胶参与辐射开关&#xff08;aer_ra_feedback&#xff09;&#xff0c;其他的实验则关闭气溶胶参与辐射过程开关&#xff0c;也有去掉某些气溶胶的影响&#xff0c;如黑碳&#xff08;BC&…

专题:2025人形机器人与服务机器人技术及市场报告|附130+份报告PDF汇总下载

原文链接&#xff1a;https://tecdat.cn/?p43583 当特斯拉Optimus在工厂里精准分拣电池&#xff0c;当普渡机器人在酒店完成跨楼层配送&#xff0c;一个万亿级的智能革命正在拉开序幕。服务机器人与人形机器人不再是实验室里的概念&#xff0c;而是正在重塑制造业、服务业的“…

JS 模块化与打包工具

一、模块化体系&#xff1a;ESM vs CJS 深入1.语法与静态性(1)ESM:静态语法&#xff0c;可被打包器做 Tree-shakingexport function play() {}export default ...import { play } from ./mod.js(2)CJS:运行时 require() , 分析能力弱&#xff0c;不利于 Tree-shaking2.Node 解析…

防御保护11

带宽管理 --- 设备对自身的流量进行管理和控制&#xff0c;去提供带宽保证、带宽限制等等功能。 带宽限制 带宽保证 连接数限制 应用场景 实现带宽管理 带宽通道 --- 定义了被管理对象所能使用的带宽资源 整体的保证带宽和最大带宽&#xff1b; SW1-SW2&#xff1a;VLAN 201 --…

[激光原理与应用-254]:理论 - 几何光学 - 自动对焦的原理

自动对焦&#xff08;Auto Focus, AF&#xff09;是现代光学系统&#xff08;如相机、手机摄像头、监控设备等&#xff09;的核心功能之一&#xff0c;其原理是通过检测成像面的清晰度或测量物体距离&#xff0c;驱动透镜组移动至最佳对焦位置。以下是自动对焦的详细原理及技术…

【Python办公】Mermaid代码转图片工具 - Tkinter GUI版本

目录 专栏导读 项目简介 功能特性 🎨 直观的图形界面 📝 代码编辑功能 🖼️ 图片生成与预览 💾 文件操作 ⚡ 性能优化 技术架构 核心技术栈 架构设计 安装与使用 环境要求 依赖安装 运行程序 使用步骤 代码示例 基本流程图 时序图 甘特图 核心代码解析 1. 主类结构 2. …

【Activiti】要点初探

Activiti 7.0.0配置 流程配置节点流程XML流程部署部署后会操作表&#xff1a;&#xff08;每部署一次增加一条记录&#xff09; ACT_RE_DEPLOYMENT 流程定义部署表 ACT_RE_PROCDEF 流程定义表 ACT_GE_BYTEARRAY 流程启动查看任务&#xff08;张三要查看准备办理任务&#xff0…

VBS 字符串处理

一. 字符串是由Unicode字符组成的一串字符。通常由数字&#xff0c;字母&#xff0c;符号组成。二. 常用函数1. 消除空格 Ltrim: 删除字符串左侧的空格。 Rtrim: 删除字符串右侧的空格。 trim: 删除字符串左侧和右侧的空格。a" hello " b"sx"msgbo…

《算法导论》第 21 章-用于不相交集合的数据结构

引言不相交集合&#xff08;Disjoint Set&#xff09;&#xff0c;也称为并查集&#xff08;Union-Find&#xff09;&#xff0c;是一种非常实用的数据结构&#xff0c;主要用于处理一些元素分组的问题。它支持高效的集合合并和元素查找操作&#xff0c;在很多算法中都有重要应…

基于51单片机RFID智能门禁系统红外人流量计数统计

1 系统功能介绍 本设计基于STC89C52单片机&#xff0c;集成RFID读卡器、红外避障传感器、继电器、LCD1602液晶显示和蜂鸣器&#xff0c;实现智能门禁与人流量统计功能。系统能够识别合法的RFID卡开门&#xff0c;并实时统计通过人数&#xff0c;具有安全报警和直观显示功能。具…

c#,vb.net全局多线程锁,可以在任意模块或类中使用,但尽量用多个锁提高效率

Public ReadOnly LockObj As New Object() 全局多线程锁 VB.NET模块中的LockObj 可以在任意模块或类中使用吧 在 VB.NET 中&#xff0c;模块&#xff08;Module&#xff09;中声明的 Public ReadOnly LockObj 可以被其他模块或类访问和使用&#xff0c;但需要注意其可见性范围…

企业安全运维服务计划书

安全运维服务计划书 一、概述 为保障企业信息系统安全、稳定、高效运行,防范各类网络安全风险,提升整体安全防护能力,特制定本安全运维服务计划书。本计划旨在通过系统化、规范化的安全运维流程,全面识别、评估、处置并持续监控企业网络环境中的安全风险,构建主动防御与…

小杰python高级(four day)——matplotlib库

1.绘制子图的方式pyplot中函数subplotFigure类中的函数add_subplotpyplot中函数subplotsfig, ax plt.subplots(nrows1, ncols1, *, sharexFalse, shareyFalse,squeezeTrue, subplot_kwNone, gridspec_kwNone, **fig_kw) 功能&#xff1a;绘制多个子图&#xff0c;可以一次生成…

C# 编程out 参数需要在函数体内部初始化,然后引用的时候无需初始化

核心规则方法内部必须初始化&#xff1a;在方法体中&#xff0c;必须在方法返回前对 out 参数显式赋值&#xff08;未赋值会导致编译错误&#xff09;调用时无需初始化&#xff1a;调用方传递 out 参数前不需要初始化变量&#xff08;可直接使用未赋值的局部变量&#xff09;下…