问题复现:
多个线程同时调用一个分页的接口和一个未分页的接口,可能导致调用原本没有使用分页的接口,查出的数据却被分页。
问题原因:
在项目中使用了PageHelper做分页,PageHelper会导致ThreadLocal的线程复用问题,从而就导致没有进行分页的接口进行了分页。
原理解释:
PageHelper会将前端传过来的参数保存到一个page对象中,然后将page的副本放到ThreadLocal中,保证多个线程之间互不影响。
如果设置了分页,即:使用了PageHelper.startPage(),那么会先运行ThreadLocal变量的set()方法,之后执行查询等逻辑,当查询逻辑正常执行完毕,最后执行ThreadLocal变量的remove()方法,将副本变量删除。
问题的关键就在于,如果在执行了ThreadLocal的set()方法成功了,但是查询等逻辑异常结束了,也就不能被执行remove()方法,那这个线程的ThreadLocal里的变量也就无法被清除,然后该线程不会被清除,而是在线程池中的状态被设置为空闲状态。
假如,有一个不需要分页的查询(会用到一个线程),去线程池中拿一个线程使用,如果恰巧用到了刚才没被清除副本变量的线程,那么就会导致查询时用到旧的变量进行分页,从而造成返回数据的错误。
解决方法参考:
使用拦截器将所有request请求进行ThreadLocal的remove()
import com.github.pagehelper.PageHelper;
import org.springframework.web.servlet.HandlerInterceptor;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 初始化PageHelper里的线程ThreadLocal
*/
public class PageLocalWebInterceptor implements HandlerInterceptor {
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
// PageHelper.clearPage() 内部调用 LOCAL_PAGE.remove()
PageHelper.clearPage();
}
}
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
/**
* 配置类:初始化PageHelper里的线程ThreadLocal
*/
@Configuration
public class FrameworkAutoConfig implements WebMvcConfigurer {
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new PageLocalWebInterceptor());
}
}