1. 为什么一定要从 dispatcherservlet 讲起
异常不是在 controller 里被“解决”的,而是在框架层被“接管”的。
在 spring mvc 中,这个“接管者”只有一个入口:
dispatcherservlet
无论你使用的是:
- @exceptionhandler
- @controlleradvice
- @responsestatus
- spring boot 默认的 /error
它们最终都必须经过 dispatcherservlet 的调度与分发。
2. dispatcherservlet 在请求中的角色定位
在一次完整的请求处理中,dispatcherservlet 的职责可以概括为四步:
- 查找 handler
- 执行 handler
- 处理返回值
- 处理异常
异常并不是一个“补丁逻辑”,而是 dispatcherservlet 的标准流程之一。
3. dodispatch:异常真正被捕获的地方
dispatcherservlet 的核心方法是 dodispatch,异常处理的关键逻辑就在这里。
3.1 dodispatch 的整体结构(简化)
protected void dodispatch(httpservletrequest request, httpservletresponse response) {
try {
// 1. 查找 handler
// 2. 执行 handler
} catch (exception ex) {
dispatchexception = ex;
} catch (throwable err) {
dispatchexception = new servletexception(err);
}
processdispatchresult(request, response, handler, dispatchexception);
}
非常重要的一点:
dispatcherservlet 并不会在 catch 中直接处理异常,而是统一交给
processdispatchresult。
3.2 throwable 为什么会被单独捕获?
catch (throwable err) {
dispatchexception = new servletexception(err);
}
这里体现了一个非常关键的设计思想:
- 框架不允许 throwable 直接向外传播
- 所有异常最终都会被“标准化”为 exception
这保证了后续异常解析链的统一性。
4. processdispatchresult:异常处理的真正入口
private void processdispatchresult(
httpservletrequest request,
httpservletresponse response,
handlerexecutionchain mappedhandler,
exception exception) {
if (exception != null) {
mv = processhandlerexception(request, response, handler, exception);
}
}
只要 exception != null,就会进入异常处理流程。
5. processhandlerexception:责任链的起点
protected modelandview processhandlerexception(
httpservletrequest request,
httpservletresponse response,
object handler,
exception ex) {
for (handlerexceptionresolver resolver : this.handlerexceptionresolvers) {
modelandview mv = resolver.resolveexception(request, response, handler, ex);
if (mv != null) {
return mv;
}
}
throw ex;
}
这一段代码,是 spring mvc 异常机制的灵魂。
从中可以明确看出三点:
- 异常处理是一个 resolver 链
- 按顺序逐个尝试解析
- 谁先返回非 null,谁就“吃掉”异常
6. handlerexceptionresolver 责任链模型
6.1 接口定义
public interface handlerexceptionresolver {
modelandview resolveexception(
httpservletrequest request,
httpservletresponse response,
object handler,
exception ex);
}
设计目的非常明确:
给异常一个“翻译成响应”的机会
6.2 默认的三个异常解析器
spring mvc 默认注册了三个 resolver:
| resolver | 作用 |
|---|---|
| exceptionhandlerexceptionresolver | 处理 @exceptionhandler |
| responsestatusexceptionresolver | 处理 @responsestatus |
| defaulthandlerexceptionresolver | 处理 spring 内置异常 |
它们构成了一条有顺序、有分工、有兜底的异常责任链。
7. resolver 链的执行顺序是如何确定的
resolver 并不是写死的,而是通过初始化流程注入:
this.handlerexceptionresolvers = getdefaultstrategies(context, handlerexceptionresolver.class);
最终顺序为:
- exceptionhandlerexceptionresolver
- responsestatusexceptionresolver
- defaulthandlerexceptionresolver
顺序的设计逻辑是:
- 用户自定义优先
- 注解语义其次
- 框架兜底最后
8. 异常是如何被“吃掉”的?
当某个 resolver 返回了非 null 的 modelandview:
if (mv != null) {
return mv;
}
意味着:
- 异常被成功解析
- 后续 resolver 不再执行
- dispatcherservlet 不会再抛出异常
这也是为什么:
一个异常只会被一个 resolver 处理
9. 如果所有 resolver 都不处理会怎样?
throw ex;
结果是:
- 异常继续向上抛
- 对 servlet 容器来说,这是一个未处理异常
- 在 spring boot 中,通常会被 /error 接管(后续篇章重点)
10. 异常责任链流程图
图1 spring mvc 异常解析责任链流程图
11. 为什么说这是一个“非常优雅的设计”
从源码可以清楚看到:
- 没有 if-else 地狱
- 没有硬编码异常类型
- 完全遵循 开闭原则
你可以:
- 插入自定义 resolver
- 调整顺序
- 替换默认行为
而 dispatcherservlet 不需要修改一行代码。
12. 本篇关键结论
到这一篇为止,我们已经明确:
- dispatcherservlet 是异常处理的唯一入口
- 异常处理不是一个方法,而是一条责任链
- @exceptionhandler 只是其中一个 resolver
- spring mvc 把“异常 → 响应”的逻辑彻底解耦
但还有几个绕不开的问题:
- @exceptionhandler 是如何被扫描并匹配异常的?
- @controlleradvice 为什么能全局生效?
- responsebody 是如何写入响应的?
- spring boot 为什么要额外引入 /error?
👉 这些问题,必须进入 resolver 内部才能解释清楚。
到此这篇关于spring boot全局异常处理机制中dispatcherservlet的处理流程和作用的文章就介绍到这了,更多相关spring boot异常处理机制dispatcherservlet内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
参考资料
- spring framework reference – exception handling
https://docs.spring.io/spring-framework/reference/web/webmvc/mvc-controller/ann-exceptionhandler.html - dispatcherservlet 源码
https://github.com/spring-projects/spring-framework
发表评论