SpringBoot中WebMvcConfigurer注册多个拦截器(addInterceptors)时的顺序问题(二)

在Spring MVC的拦截器(Interceptor)中,preHandlepostHandleafterCompletion 这三个方法的执行顺序与整个请求-响应流程密切相关。以下是它们的详细执行顺序和触发时机:


1. 完整执行顺序流程图

复制代码
客户端请求 → 
    Interceptor1.preHandle → 
        Interceptor2.preHandle → 
            Controller方法执行 → 
                Interceptor2.postHandle → 
                    Interceptor1.postHandle → 
                        视图渲染(如果有) → 
                            Interceptor2.afterCompletion → 
                                Interceptor1.afterCompletion → 
                                    响应返回客户端

2. 各方法的具体作用与顺序

(1) preHandle(请求到达Controller前)
  • 执行顺序 :按拦截器的order从小到大 依次执行(order值越小优先级越高)。

  • 返回值

    • 返回 true:继续调用后续拦截器或Controller。

    • 返回 false:中断请求,直接返回(后续拦截器、Controller、postHandleafterCompletion均不会执行)。

  • 典型用途:权限校验、参数预处理、日志记录。

(2) postHandle(Controller执行后,视图渲染前)
  • 执行顺序 :按拦截器的order从大到小 依次执行(与preHandle相反)。

  • 触发时机:Controller方法执行完毕,但尚未进行视图渲染(如Thymeleaf、Freemarker等)。

  • 注意 :如果Controller中发生异常,此方法不会被执行

  • 典型用途:修改ModelAndView、记录响应数据。

(3) afterCompletion(请求完成后,包括视图渲染)
  • 执行顺序 :按拦截器的order从大到小 依次执行(与postHandle相同)。

  • 触发时机:整个请求处理完成(包括视图渲染),无论是否有异常都会执行。

  • 典型用途:资源清理、性能监控、异常日志记录。


3. 关键注意事项

  1. 拦截器链的执行

    • 若某个拦截器的preHandle返回false,后续拦截器的preHandle、Controller、postHandleafterCompletion均不会执行。

    • 只有所有preHandle都返回true时,Controller才会执行。

  2. 异常处理的影响

    • 如果Controller抛出异常:

      • postHandle不会执行

      • afterCompletion仍会执行 (可通过ex参数获取异常对象)。

  3. @ControllerAdvice的差异

    • 拦截器的afterCompletion在视图渲染后触发,而@ControllerAdvice@ExceptionHandler在异常发生时立即处理(早于afterCompletion)。

4. 示例场景

假设有两个拦截器:

  • AuthInterceptororder = 1

  • LoggingInterceptororder = 2

正常流程:
复制代码
1. AuthInterceptor.preHandle()     (order=1, 最先执行)
2. LoggingInterceptor.preHandle() (order=2)
3. Controller方法执行
4. LoggingInterceptor.postHandle() (order=2, 逆序)
5. AuthInterceptor.postHandle()   (order=1)
6. 视图渲染
7. LoggingInterceptor.afterCompletion() (order=2, 逆序)
8. AuthInterceptor.afterCompletion()   (order=1)
中断流程(如权限校验失败):
复制代码
1. AuthInterceptor.preHandle() 返回false
   // 后续所有方法均不执行,直接返回客户端
异常流程(Controller抛出异常):
复制代码
1. AuthInterceptor.preHandle()     (order=1)
2. LoggingInterceptor.preHandle()  (order=2)
3. Controller抛出异常
4. LoggingInterceptor.afterCompletion() (携带异常对象)
5. AuthInterceptor.afterCompletion()   (携带异常对象)
   // postHandle 不会执行

5. 总结

  • preHandle:是拦截器的"入口",决定请求是否继续。

  • postHandle:可修改响应数据,但无法处理异常。

  • afterCompletion:最适合做资源清理和最终日志记录,无论成功失败都会执行。

通过合理组合这三个方法,可以实现灵活的请求处理逻辑(如权限链、日志追踪、性能监控等)。

相关推荐
森林里的程序猿猿几秒前
Spring Aop底层源码实现(一)
java·后端·spring
while(1){yan}4 分钟前
个人抽奖系统测试报告
spring boot·java-ee·压力测试
耗子会飞7 分钟前
小白学习springboot项目如何连接RocketMQ
后端·rocketmq
ZTrainWilliams14 分钟前
swagger-mcp-toolkit 让 AI编辑器 更快“读懂并调用”你的接口
前端·后端·mcp
cylgdzz11127 分钟前
PageIndex:一种不靠向量检索的长文档 RAG 实现思路
后端
weixin_4563216430 分钟前
Java架构设计:Redis持久化方案整合实战
java·开发语言·redis
Later33 分钟前
Apache Doris 深度讲解:从核心概念到实战项目
后端
攒了一袋星辰40 分钟前
SequenceGenerator高并发有序顺序号生成中间件 - 架构设计文档
java·后端·spring·中间件·架构·kafka·maven
码农刚子43 分钟前
字符串拼接用“+”还是 StringBuilder?别再凭感觉写了
后端·代码规范
lzp07911 小时前
SpringBoot3.3.0集成Knife4j4.5.0实战
java