可以用 request 携带数据。
更优雅的方式是用threadlocal
。请求进入tomcat 和产生响应前,都处于同一个线程中
比如在一个登录拦截器中,在preHandle
方法中登录成功后,放行前,想把user对象传到controller或service中
- 创建一个类
UserThreadLocal.java
public class UserThreadLocal {
//把构造函数私有,外面不能new,只能通过下面两个方法操作
private UserThreadLocal(){
}
private static final ThreadLocal<User> LOCAL = new ThreadLocal<User>();
public static void set(User user){
LOCAL.set(user);;
}
public static User get(){
return LOCAL.get();
}
}
相当于一个容器,此容器伴随着线程,线程启动,就有这个容器,销毁,容器就跟着销毁。生命周期就是这个线程。
- 在拦截器中,登录成功后,放行前加上
//登录成功
UserThreadLocal.set(user);//将user对象放置在本地线程中,方便controller和service获取
由于tomcat 的运行机制,要及时清空threadLocal的内容
以下可以放在拦截器的afterCompletion方法中
/*tomcat底层 每一个请求都是一个线程,如果每一个请求都启动一个线程,性能就会降低,
1. 于是就有了线程池,而线程池中的线程并不是真正销毁或真正启动的。
2. 也就是说这个请求的线程是个可复用的线程,第二次请求可能还会拿到刚刚的线程,
3. 若不清空,里面本身就有user对象,数据会错乱*/
UserThreadLocal.set(null); //清空本地线程中的user对象数据
- 在controller 或 service 中调用
User user = UserThreadLocal.get();