聊聊如何让springboot拦截器的执行顺序按我们想要的顺序执行
前言
最近朋友和我提了一个挺有趣的问题:他们有个项目用了他们框架部提供的jwt token校验填充组件,实现原理大概是,通过springboot拦截器来校验token,如果token合法,就解析token,将token携带的业务信息map填充到threadlocal里面,方便后续业务使用。
朋友的问题就是他想往这个threalocal里面的业务map再扩展一些业务字段,但因为这个组件不是朋友的部门开发的,他就不能改源码,只能通过扩展的方式。
他的思路就是他也写一个拦截器,在这个拦截器里面做业务填充。这边有个前提就是框架部的执行时机得在朋友写的拦截器之前,朋友的做法是在他写的拦截器上面加@Order注解,不过发现不管用。于是就找我讨论一下这个问题。
抽象出来的问题就是标题说的如何让springboot拦截器的执行顺序按我们想要的顺序执行
思路
方法一:自己的业务项目写一个和框架组一模一样的类
即这个类和框架组提供的包名和类名一样,然后改这个类,这个实现原理是利用了类的加载顺序
方法二:利用org.springframework.web.servlet.config.annotation.InterceptorRegistration#order()
不过这个order方法是spring 4.3+版本后才提供。
具体使用形如下
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(helloHandlerInterceptor).addPathPatterns("/**").order(100);
registry.addInterceptor(otherHelloHandlerInterceptor).addPathPatterns("/**").order(-1);
}
通过配置order()的值,值越小,优先级越高。不配默认是0
那为啥要配置这个呢,如果对springmvc有稍微深入一下的话,拦截器链最终是会用到
protected List<Object> getInterceptors() {
return this.registrations.stream()
.sorted(INTERCEPTOR_ORDER_COMPARATOR)
.map(InterceptorRegistration::getInterceptor)
.collect(Collectors.toList());
}
排序就是根据这个order来的
总结
本文提供的方案二适用于spring 4.3+版本,低于该版本,请慎重。
demo链接
https://github.com/lyb-geek/springboot-learning/tree/master/springboot-interceptor-order