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的问题提问,而不仅仅只限于项目
- 针对性服务:有没理解的地方,文档或者视频还没有讲到可以提出,本人会补充
- 面试与简历指导:提供面试回答技巧,项目怎样写才能在简历中具有独特的亮点
- 中间件环境:对于项目中需要使用的中间件,可直接替换成我提供的云环境
- 面试后复盘:小伙伴去面试后,如果哪里被面试官问住了,可以再找我解答
- 远程的解决:如果在启动项目遇到问题,本人可以帮你远程解决
进入星球后,即可享受上述所有服务,保证不会再有其他隐藏费用。
