Java常用异常处理类库与Exception

发布时间 - 2026-01-06 00:00:00    点击率:
Exception是程序应处理的异常(如IOException),Error是JVM级严重问题(如OutOfMemoryError)不应捕获;RuntimeException子类为unchecked异常,需逻辑规避;checked异常必须显式处理。

Java里ExceptionError到底怎么分?

不是所有“出错了”都该捕获——Exception是程序能(且应该)处理的异常,比如IOExceptionSQLException;而Error(如OutOfMemoryErrorStackOverflowError)代表JVM级严重问题,一般不捕获、不重试、不掩盖。

常见误操作:用catch (Exception e)兜底所有异常,结果把本该崩溃的NullPointerException吞掉,掩盖了空指针源头;或试图捕获NoClassDefFoundError并“恢复”,实际类加载失败已不可逆。

  • RuntimeException及其子类(如IllegalArgumentExceptionConcurrentModificationException)属于unchecked异常,编译器不强制处理,但应通过逻辑校验提前规避,而非依赖try-catch
  • 必须显式声明或捕获的是checked异常(即Exception直接子类,但非RuntimeException),比如FileNotFoundException调用new FileInputStream(path)时必须处理
  • 自定义异常建议继承RuntimeException(业务异常)或Exception(需强制调用方关注的外部异常,如支付网关超时)

Apache Commons Lang的ExceptionUtils怎么用才不踩坑?

它最常被用来获取异常完整堆栈、查找特定类型异常、判断是否嵌套等,但默认行为容易误导:

  • ExceptionUtils.getStackTrace(e)返回字符串,含换行符,日志中直接打印可能破坏结构化日志格式,建议用log.error("msg", e)代替手动拼接
  • ExceptionUtils.indexOfType(e, SQLException.class)只查直接类型,不递归Cause链;要查整个异常链是否含某类,得用ExceptionUtils.containsType(e, SQLException.class)
  • 注意getCause()可能为null,而ExceptionUtils.getRootCause(e)会安全遍历到最内层非null cause,但若中间有循环引用(极少见),会抛IllegalStateException
ExceptionUtils.getRootCause(new SQLException("outer", new RuntimeException("inner")))

返回的是RuntimeException("inner"),不是SQLException本身。

立即学习“Java免费学习笔记(深入)”;

Spring的@ExceptionHandler为什么有时不生效?

它只对当前Controller内抛出的异常生效,且受异常传播路径限制:

  • 异步方法(@Async)中抛出的异常不会到达Controller的@ExceptionHandler,因为执行线程已切换,需单独配置AsyncUncaughtExceptionHandler
  • 过滤器(Filter)或拦截器(Interceptor)中抛出的异常,绕过Spring MVC异常解析流程,@ExceptionHandler捕获不到,应改用@ControllerAdvice配合@ExceptionHandler全局处理,或在Filter中自行try-catch
  • 如果异常被更上层AOP切面(如事务管理器)提前捕获并转为其他异常(如TransactionSystemException),原异常类型丢失,@ExceptionHandler按原类型写的就不会匹配

Logback/SLF4J记录异常时,logger.error("msg", e)logger.error("msg: " + e)区别很大

前者把Exception对象作为独立参数传入,日志框架能自动提取堆栈并结构化输出;后者只是字符串拼接,堆栈信息完全丢失,只剩e.toString()(通常是类名+message)。

  • 拼接方式还可能触发不必要的toString()getMessage()调用,在异常未初始化好时导致NullPointerException
  • 使用占位符(logger.error("Failed to process {}", id, e))是安全的:只有当日志级别允许输出时,才会计算参数,避免无谓的对象创建或toString开销
  • 敏感信息(如密码、token)不要出现在getMessage()里,因为堆栈日志默认包含它;应把敏感数据存入异常的suppressed或自定义字段,再通过MDC或自定义ThrowableProxy过滤

异常链越深,越要确认每一层是否真的需要包装——过度嵌套会让根因难定位,而完全不包装又丢失上下文。平衡点通常在:底层IO/DB异常保留原始类型,业务层用自定义BusinessException封装,并通过initCause()关联原始异常。


# java  # apache  #   # ai  # proxy  # stream  # 区别  # spring mvc  # 敏感数据  # overflow 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?  php中::能调用final静态方法吗_final修饰静态方法调用规则【解答】  PHP的CURL方法curl_setopt()函数案例介绍(抓取网页,POST数据)  如何用AWS免费套餐快速搭建高效网站?  java获取注册ip实例  手机怎么制作网站教程步骤,手机怎么做自己的网页链接?  Laravel如何使用Livewire构建动态组件?(入门代码)  Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法  青岛网站建设如何选择本地服务器?  网站制作大概要多少钱一个,做一个平台网站大概多少钱?  如何快速选择适合个人网站的云服务器配置?  PythonWeb开发入门教程_Flask快速构建Web应用  制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?  Laravel如何构建RESTful API_Laravel标准化API接口开发指南  Laravel如何生成URL和重定向?(路由助手函数)  如何在建站之星绑定自定义域名?  Laravel怎么解决跨域问题_Laravel配置CORS跨域访问  Laravel如何使用Facades(门面)及其工作原理_Laravel门面模式与底层机制  利用vue写todolist单页应用  Laravel中间件如何使用_Laravel自定义中间件实现权限控制  如何快速配置高效服务器建站软件?  如何在云主机快速搭建网站站点?  微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】  JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  Laravel怎么写单元测试_PHPUnit在Laravel项目中的基础测试入门  如何基于云服务器快速搭建网站及云盘系统?  Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理  Laravel怎么实现模型属性的自动加密  Laravel集合Collection怎么用_Laravel集合常用函数详解  Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】  装修招标网站设计制作流程,装修招标流程?  php 三元运算符实例详细介绍  jimdo怎样用html5做选项卡_jimdo选项卡html5实现与切换效果【指南】  Firefox Developer Edition开发者版本入口  Laravel如何实现模型的全局作用域?(Global Scope示例)  如何在香港服务器上快速搭建免备案网站?  Laravel如何自定义分页视图?(Pagination示例)  Python并发异常传播_错误处理解析【教程】  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  Python进程池调度策略_任务分发说明【指导】  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  怎么用AI帮你为初创公司进行市场定位分析?  在centOS 7安装mysql 5.7的详细教程  如何在IIS7中新建站点?详细步骤解析  Laravel中DTO是什么概念_在Laravel项目中使用数据传输对象(DTO)  如何快速查询域名建站关键信息?  Swift中循环语句中的转移语句 break 和 continue  edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】  Laravel如何创建自定义Artisan命令?(代码示例)