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

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

相关推荐
涡能增压发动积20 小时前
同样的代码循环 10次正常 循环 100次就抛异常?自定义 Comparator 的 bug 让我丢尽颜面
后端
云烟成雨TD20 小时前
Spring AI Alibaba 1.x 系列【6】ReactAgent 同步执行 & 流式执行
java·人工智能·spring
Wenweno0o20 小时前
0基础Go语言Eino框架智能体实战-chatModel
开发语言·后端·golang
于慨20 小时前
Lambda 表达式、方法引用(Method Reference)语法
java·前端·servlet
swg32132120 小时前
Spring Boot 3.X Oauth2 认证服务与资源服务
java·spring boot·后端
tyung20 小时前
一个 main.go 搞定协作白板:你画一笔,全世界都看见
后端·go
gelald20 小时前
SpringBoot - 自动配置原理
java·spring boot·后端
@yanyu66620 小时前
07-引入element布局及spring boot完善后端
javascript·vue.js·spring boot
殷紫川20 小时前
深入理解 AQS:从架构到实现,解锁 Java 并发编程的核心密钥
java
一轮弯弯的明月20 小时前
贝尔数求集合划分方案总数
java·笔记·蓝桥杯·学习心得