跳到主要内容

ThreadLocal使用中致命的细节问题

我们在之前已经讲了很多关于 ThreadLocal 的设计与多线程的解决方案,而本章节介绍一个使用中容易被忽略的细节,保证你看完了后直拍大腿

BaseParameterHolder

先设计一个ThreadLocal的工具

public class BaseParameterHolder {

private static final ThreadLocal<Map<String, String>> THREAD_LOCAL_MAP = new ThreadLocal<>();


public static void setParameter(String name, String value) {
Map<String, String> map = THREAD_LOCAL_MAP.get();
if (map == null) {
map = new HashMap<>(64);
}
map.put(name, value);
THREAD_LOCAL_MAP.set(map);
}

public static String getParameter(String name) {
return Optional.ofNullable(THREAD_LOCAL_MAP.get()).map(map -> map.get(name)).orElse(null);
}

public static void removeParameter(String name) {
Map<String, String> map = THREAD_LOCAL_MAP.get();
if (map != null) {
map.remove(name);
}
}

public static ThreadLocal<Map<String, String>> getThreadLocal() {
return THREAD_LOCAL_MAP;
}

public static Map<String, String> getParameterMap() {
Map<String, String> map = THREAD_LOCAL_MAP.get();
if (map == null) {
map = new HashMap<>(64);
}
return map;
}

public static void setParameterMap(Map<String, String> map) {
THREAD_LOCAL_MAP.set(map);
}

public static void removeParameterMap(){
THREAD_LOCAL_MAP.remove();
}
}

BaseParameterHolder 中有个 ThreadLocal 类型成员属性 THREAD_LOCAL_MAP,里面放的数据是Map结构。

在操作 BaseParameterHolder 进行设值和取值的时候,其实就是操作 THREAD_LOCAL_MAP 中的Map结构

看着很简单,接下来用案例来详细解释

案例1

public class Test01 {

public static void main(String[] args) {
BaseParameterHolder.setParameter("name", "小红");
ExecutorService executors = Executors.newFixedThreadPool(10);

Map<String, String> parameterMap = BaseParameterHolder.getParameterMap();
System.out.println("主线程执行的数据获取:" + BaseParameterHolder.getParameter("name"));
executors.execute(() -> {
try {
Thread.sleep(500);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
BaseParameterHolder.setParameterMap(parameterMap);
System.out.println("线程池执行的数据获取:" + BaseParameterHolder.getParameter("name"));
});
}
}

先在主线程中,向BaseParameterHolder中存放数据,然后将数据取出,进行输出。

在线程池执行中,再把数据放入BaseParameterHolder中,然后将数据取出,进行输出。

结果

主线程执行的数据获取:小红
线程池执行的数据获取:小红

这样做肯定主线程和子线程都能取到数据

案例2

一般在操作 ThreadLocal 时,为了防止内存泄露,通常会在使用完后,手动的删除,所以在案例2中多了个删除的动作

public class Test02 {

public static void main(String[] args) {
BaseParameterHolder.setParameter("name", "小红");
ExecutorService executors = Executors.newFixedThreadPool(10);

Map<String, String> parameterMap = BaseParameterHolder.getParameterMap();
System.out.println("主线程执行的数据获取:" + BaseParameterHolder.getParameter("name"));
executors.execute(() -> {
try {
Thread.sleep(500);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
BaseParameterHolder.setParameterMap(parameterMap);
System.out.println("线程池执行的数据获取:" + BaseParameterHolder.getParameter("name"));
});
System.out.println("主线程执行删除数据");
BaseParameterHolder.removeParameter("name");
}
}

大家可以先思考一下结果是什么,然后再看结果

付费内容提示

该文档的全部内容仅对「JavaUp项目实战&技术讲解」知识星球用户开放

加入星球后,你可以获得:

  • 超级八股文:100万+字的全栈技术知识库,涵盖技术核心、数据库、中间件、分布式等深度剖析的讲解
  • 讲解文档:黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的从0到1的550+详细文档
  • 讲解视频:黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的核心业务详细讲解
  • 1 对 1 解答:可以对我进行1对1的问题提问,而不仅仅只限于项目
  • 针对性服务:有没理解的地方,文档或者视频还没有讲到可以提出,本人会补充
  • 面试与简历指导:提供面试回答技巧,项目怎样写才能在简历中具有独特的亮点
  • 中间件环境:对于项目中需要使用的中间件,可直接替换成我提供的云环境
  • 面试后复盘:小伙伴去面试后,如果哪里被面试官问住了,可以再找我解答
  • 远程的解决:如果在启动项目遇到问题,本人可以帮你远程解决
进入星球后,即可享受上述所有服务,保证不会再有其他隐藏费用。
知识星球二维码

1. 打开微信 -> 扫描左侧二维码 -> 加入「JavaUp项目实战&技术讲解」知识星球

2. 查看星球使用指导,获取完整项目讲解资料索引

👉 点击解锁全部付费内容
🎁优惠