前言

最近敌人和我提了一个挺乏味的问题:他们有个我的项目用了他们框架部提供的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