利用threadLocal 把拦截器中的对象传递到controller或service中

可以用 request 携带数据。

更优雅的方式是用threadlocal。请求进入tomcat 和产生响应前,都处于同一个线程中

比如在一个登录拦截器中,在preHandle方法中登录成功后,放行前,想把user对象传到controller或service中

  1. 创建一个类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();
    }
}

相当于一个容器,此容器伴随着线程,线程启动,就有这个容器,销毁,容器就跟着销毁。生命周期就是这个线程。

  1. 在拦截器中,登录成功后,放行前加上
 //登录成功
UserThreadLocal.set(user);//将user对象放置在本地线程中,方便controller和service获取

由于tomcat 的运行机制,要及时清空threadLocal的内容

以下可以放在拦截器的afterCompletion方法中

/*tomcat底层 每一个请求都是一个线程,如果每一个请求都启动一个线程,性能就会降低,
 1. 于是就有了线程池,而线程池中的线程并不是真正销毁或真正启动的。
 2. 也就是说这个请求的线程是个可复用的线程,第二次请求可能还会拿到刚刚的线程,
 3. 若不清空,里面本身就有user对象,数据会错乱*/
UserThreadLocal.set(null); //清空本地线程中的user对象数据
  1. 在controller 或 service 中调用
User user = UserThreadLocal.get();
拦截器使用 `ThreadLocal` 存储数据时,需要注意以下几点: 1. `ThreadLocal` 的作用域是线程级别的,即同一个线程的所有代码都可以访问该 `ThreadLocal` 对象存储的数据,但不同线程之间是互不干扰的。 2. 在拦截器,如果使用 `ThreadLocal` 存储数据,需要在请求处理结束后将其清空,否则该线程在下一次请求时可能会访问到上一次请求存储的数据。 3. 在控制器访问 `ThreadLocal` 存储的数据时,需要确保在同一个线程,否则会出现获取到 `null` 的情况。 以下是一个示例代码: ```java public class MyInterceptor implements HandlerInterceptor { private static final ThreadLocal<String> threadLocal = new ThreadLocal<>(); @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String value = // 获取需要存储的数据 threadLocal.set(value); return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { threadLocal.remove(); // 清空 ThreadLocal 的数据 } } @RestController public class MyController { @GetMapping("/test") public String test() { String value = MyInterceptor.threadLocal.get(); return "value: " + value; } } ``` 在上面的代码,`MyInterceptor` 使用 `ThreadLocal` 存储数据,`MyController` 通过访问 `MyInterceptor.threadLocal` 获取存储的数据。请注意,`MyInterceptor.threadLocal` 是一个静态变量,因此在不同的线程访问到的是不同的 `ThreadLocal` 对象。如果在 `MyController` 访问时获取到的是 `null`,那么很可能是因为在不同的线程访问了 `ThreadLocal` 对象
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值