当前位置: 代码网 > it编程>编程语言>Java > Spring Boot全局异常处理机制中DispatcherServlet的处理流程和作用

Spring Boot全局异常处理机制中DispatcherServlet的处理流程和作用

2026年01月31日 Java 我要评论
1. 为什么一定要从 dispatcherservlet 讲起异常不是在 controller 里被“解决”的,而是在框架层被“接管”的。在 sprin

1. 为什么一定要从 dispatcherservlet 讲起

异常不是在 controller 里被“解决”的,而是在框架层被“接管”的。

在 spring mvc 中,这个“接管者”只有一个入口:

dispatcherservlet

无论你使用的是:

  • @exceptionhandler
  • @controlleradvice
  • @responsestatus
  • spring boot 默认的 /error

它们最终都必须经过 dispatcherservlet 的调度与分发。

2. dispatcherservlet 在请求中的角色定位

在一次完整的请求处理中,dispatcherservlet 的职责可以概括为四步:

  1. 查找 handler
  2. 执行 handler
  3. 处理返回值
  4. 处理异常

异常并不是一个“补丁逻辑”,而是 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 异常机制的灵魂

从中可以明确看出三点:

  1. 异常处理是一个 resolver 链
  2. 按顺序逐个尝试解析
  3. 谁先返回非 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);

最终顺序为:

  1. exceptionhandlerexceptionresolver
  2. responsestatusexceptionresolver
  3. 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. 本篇关键结论

到这一篇为止,我们已经明确:

  1. dispatcherservlet 是异常处理的唯一入口
  2. 异常处理不是一个方法,而是一条责任链
  3. @exceptionhandler 只是其中一个 resolver
  4. 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

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2026  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com