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

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

相关推荐
fatsheep洋1 分钟前
XSS-DOM-1
java·前端·xss
七七软件开发32 分钟前
一对一交友小程序 / APP 系统架构分析
java·python·小程序·系统架构·php
TDengine (老段)38 分钟前
TDengine 中 TDgpt 异常检测的数据密度算法
java·大数据·算法·时序数据库·iot·tdengine·涛思数据
YuTaoShao1 小时前
【LeetCode 热题 100】155. 最小栈
java·算法·leetcode
程序视点1 小时前
Java语言核心特性全解析:从面向对象到跨平台原理
java·后端·java ee
Warren981 小时前
MySQL查询语句详解
java·开发语言·数据库·mysql·算法·蓝桥杯·maven
懋学的前端攻城狮1 小时前
深入浅出Linux-01:系统化掌握基础操作
linux·后端
丶小鱼丶1 小时前
Spring之【循环引用】
java·spring
考虑考虑1 小时前
Redis8中的布隆过滤器
redis·后端·程序员
hqxstudying2 小时前
Java向量化
java·开发语言