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:最适合做资源清理和最终日志记录,无论成功失败都会执行。

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

相关推荐
Mr -老鬼4 分钟前
功能需求对前后端技术选型的横向建议
开发语言·前端·后端·前端框架
IT=>小脑虎4 分钟前
Go语言零基础小白学习知识点【基础版详解】
开发语言·后端·学习·golang
程序猿阿伟5 分钟前
《Python复杂结构静态分析秘籍:递归类型注解的深度实践指南》
java·数据结构·算法
Eric_见嘉37 分钟前
NestJS 🧑‍🍳 厨子必修课(九):API 文档 Swagger
前端·后端·nestjs
黑白极客38 分钟前
怎么给字符串字段加索引?日志系统 一条更新语句是怎么执行的
java·数据库·sql·mysql·引擎
爬山算法42 分钟前
Hibernate(32)什么是Hibernate的Criteria查询?
java·python·hibernate
码农水水1 小时前
中国邮政Java面试:热点Key的探测和本地缓存方案
java·开发语言·windows·缓存·面试·职场和发展·kafka
a程序小傲1 小时前
小红书Java面试被问:TCC事务的悬挂、空回滚问题解决方案
java·开发语言·人工智能·后端·python·面试·职场和发展
短剑重铸之日1 小时前
《SpringBoot4.0初识》第五篇:实战代码
java·后端·spring·springboot4.0
heartbeat..1 小时前
Spring MVC 全面详解(Java 主流 Web 开发框架)
java·网络·spring·mvc·web