ThreadLocal模式的理解

工作也2年了,看了很多人的博客,也看了些书,都建议做技术的能够养成写博客的习惯,所以决定尝试开始写博客,可能一开始只是作为自己的知识笔记记录,希望在以后能够达到博客文的水平。


ThreadLocal的作用个人理解是使数据在不同编程层次得到有效的共享,对执行逻辑和执行数据进行有效的解耦。


ThreadLocal常用于多线程问题。


ThreadLocal类在维护变量时,实际使用了Thread中的一个叫做ThreadLocalMap的独立副本,这个ThreadLocalMap是ThreadLocal类中的一个静态类。每个线程只能修改属于自己副本中的数据。所以隔离了线程。


ThreadLocal常使用的方法使get、set,源码如下:


  public T get() {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null)
                return (T)e.value;
        }
        return setInitialValue();
    }


其中getMap()就是得到ThreadLocalMap,实则是一个默认大小为16的entry数组;

由源码可知,get()方法是得到当前线程,在用当前线程作为key 去ThreadLocalMap去找对应的value;


  public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }

由源码可知,set()方法是得到当前线程,在用当前线程作为key 往ThreadLocalMap放入value;


ThreadLocal线程安全原理:
ThreadLocalMap变量属于线程内部属性
1.线程与线程之间的数据访问隔离
2.同一线程中,不同ThreadLocal实例操作对象之间互相隔离


ThreadLocal模式的主要两个步骤:
1.建立一个类,并在其中封装一个静态的ThreadLocal变量,使其成为一个共享数据环境;
2.在类中实现访问静态ThreadLocal变量的静态方法(get、set)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Feign 是一个基于 Java 的 HTTP 客户端库,用于与 RESTful 服务进行交互。在 Feign 中,如果需要在请求之间传递数据,通常不推荐直接使用 ThreadLocal,因为 Feign 的请求处理是异步的,每个请求在单独的线程中执行。然而,如果你确实遇到了某种场景需要跨请求共享信息,比如认证令牌,可以考虑以下方法: 1. **使用装饰者模式**:你可以创建一个自定义的 `Interceptor`(拦截器),在每个请求开始前设置 ThreadLocal,然后在拦截器链的回调中获取这个值。 ```java public class MyInterceptor implements ClientRequestInterceptor { private final ThreadLocal<String> threadLocal; public MyInterceptor(String value) { threadLocal = new ThreadLocal<>(); threadLocal.set(value); } @Override public void apply(RequestTemplate requestTemplate) { // 在这里添加你需要的操作,如设置请求头等 // requestTemplate.header("Authorization", threadLocal.get()); } } ``` 使用时,将 `MyInterceptor` 添加到 Feign client 的拦截器列表中: ```java Feign.builder() .client(new OkHttpClient()) .interceptors(Arrays.asList(new MyInterceptor("your_value"))) .target(MyService.class, "http://example.com"); ``` 2. **使用响应拦截器**:对于返回的响应,你也可以设置一个全局的响应拦截器来处理 ThreadLocal 的值。 ```java public class ResponseInterceptor implements ClientResponseInterceptor { @Override public ClientResponse intercept(ClientRequest originalRequest, byte[] originalBody, Response response) { String value = threadLocal.get(); // 对响应进行处理,例如修改响应体或添加头部 return new ClientResponse(response.request(), response.response(), response.headers(), originalBody, new ByteArrayInputStream(response.body())); } } ``` 然后添加到客户端配置: ```java Feign.builder() .client(new OkHttpClient()) .responseInterceptors(Arrays.asList(new ResponseInterceptor())) .target(MyService.class, "http://example.com"); ``` 但是需要注意的是,这种做法可能会导致代码难以理解和维护,因为它绕过了 Feign 的初衷和设计。在大多数情况下,应该优先考虑使用标准的参数传递方式,如请求头、URL 查询参数或请求体。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值