多线程如果不理解透彻, 那么 ThreadLocal 始终是有些会有所迷糊的。
ThreadLocal 本身的命名有有问题, 这些美国精英整出来的技术,再加上一个奇怪的命名。对我们中国人来说,就是一场场的灾难。
如下的问题, 你觉得输出是多少呢?
/**
* Created by lk on 2017/6/8.
*/
public class ThreadLocalTest extends Thread { ThreadLocal<Long> threadLocal = new ThreadLocal<Long>() {
@Override
protected Long initialValue() {
return 2L;
}
}; public ThreadLocalTest() {
setThreadLocal(new ThreadLocal<Long>());
} @Override
public void run() {
show();
} public void setThreadLocal(ThreadLocal<Long> threadLocal) {
this.threadLocal = threadLocal;
threadLocal.set(10L);
} public void show() {
System.out.println( threadLocal.get());
} public static void main(String[] args) {
ThreadLocalTest ThreadLocalTest = new ThreadLocalTest(); ThreadLocalTest.run();
ThreadLocalTest.start();
}
}

答案是

10

null

这里有几个陷阱。

首先,ThreadLocalTest.run(); 这行由于要启动一个新的线程 以及它要初始化一个map(ThreadLocal 内部的东西)等, 它执行show 方法的时间通常会ThreadLocalTest.start();  所以其实是 ThreadLocalTest.run();  打印了null, 而 ThreadLocalTest.start(); 输出了 10。 就是说 后者比前者先输出。我们 把show 方法改成下面的样子:

System.out.println(" local = " + Thread.currentThread().getName() + "     " + threadLocal.get());
就会得到:

local = main 10
local = Thread-0 null

其次,ThreadLocalTest.run();  和 ThreadLocalTest.start();  方法看起来是执行了一样的代码, 其实不然, 这就是ThreadLocal 的非常的令人迷惑的地方。  要理解ThreadLocal , 关键在于理解其 get / set 方法, ThreadLocal  和 Thread 的关系, ThreadLocal  和 ThreadLocalMap 的关系 。一定要理清。  我们可以知道。 这里有两个线程, 一个是 main, 一个是 ThreadLocalTest。 

main 线程的执行路径是: new 一个  ThreadLocalTest ,new 的过程中, 修改了当前类变量 threadLocal(其初始值是2L),让其指向了一个新的 ThreadLocal , 其没有初始值,也就是 null (查看源码可知)。 然后又将 其值设置为 10L。  这里一定要搞清楚, 最开始的那个 threadLocal 变量的堆实例已经无法获取了, 已经变成垃圾了, 随时可能被GC了, 现在的threadLocal 变量仅仅是一个 引用, 其指向的实例 已经不是原来那个了, 初始值已经变成了 null了。threadLocal.set(10L); 改成  this.threadLocal.set(10L); 是没有任何影响的, 他们是同一个实例。

至此, 我们应该明白了, main线程对应的 ThreadLocalMap 实例存放的 threadLocal 对应的 key 的 值 是 10L, 故 main 的start 方法调用的 show , get 到的 threadLocal 的值 就是 10。

ThreadLocalTest 线程的执行路径 是明显不同的, ThreadLocalTest 只管启动一个线程, 然后执行 run 方法,run 方法调用的 show , get 到的 threadLocal 的值 是多少呢? 我们先看一下 get 的源码比较好:

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

ThreadLocalTest 线程执行到 ThreadLocalMap map = getMap(t);  获取的 map 一定是 null, 因为ThreadLocalTest 线程 是刚刚创建的,其 对应的 ThreadLocalMap 一定是 null ( 查看 Thread 的源码可知),  这样, ThreadLocalMap 立即执行 return setInitialValue(); 尝试设置 初始值,然后返回初始值。 setInitialValue 方法 其实主要是获取了  initialValue 方法的值, 然后设置到了 ThreadLocalMap 中去。  前面的分析我们得知,现在的 threadLocal 已经被悄悄改变了, 它没有复写initialValue 方法, 其初始值就是默认值, 就是null。 故 get 的结果 就是 null 。 至此, 分析完毕。

欢迎热爱底层技术的人和我一起遨游技术的海洋。还有不懂的请留言。

  

详解一个ThreadLocal 的谜题的更多相关文章

  1. ThreadLocal详解,ThreadLocal源码分析,ThreadLocal图解

    本文脉路: 概念阐释 ---->  原理图解  ------> 源码分析 ------>  思路整理  ----> 其他补充. 一.概念阐述. ThreadLocal 是一个为 ...

  2. java多线程详解(5)-Threadlocal用法

    ThreadLocal是什么 早在JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal为解决多线程程序的并发问题提供了一种新的思路. 使用这个工具类可以很简洁 ...

  3. 详解一个自己原创的正则匹配IP的表达式

    这里给大家详细讲解一下一个匹配IP地址的正则表达式, 有关正则方面的知识,会在详细的讲解中提到. 在讲解之前,我先给大家介绍一下,ip地址的生成规则. IP地址,是由32位数字二进制转为四个十进制的字 ...

  4. pc端的企业网站(IT修真院test9)详解一个响应式完成的pc端项目

    一:引入bootstrap框架 昨天一直被bootstrap栅格系统折磨. why? 我本来想一边码字,一边学习栅格布局的.but不成功.这时我头脑已经昏了. 下午,我查看了bootstrap的官网, ...

  5. Android开发——Android的消息机制详解

    )子线程默认是没有Looper的,Handler创建前,必须手动创建,否则会报错.通过Looper.prepare()即可为当前线程创建一个Looper,并通过Looper.loop()来开启消息循环 ...

  6. Drawable实战解析:Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)

    Android XML shape 标签使用详解   一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...

  7. iOS开发技巧系列---详解KVC(我告诉你KVC的一切)

    KVC(Key-value coding)键值编码,单看这个名字可能不太好理解.其实翻译一下就很简单了,就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值.而不需 ...

  8. 13 SQLiteOpenHelper SQLiteDatabase详解

    创建数据库: 1. 创建一个类继承SQLiteOpenHelper 2. 创建继承对象 new SQLiteOpenHelper() 3. 用创建的对象获取可写或者可读的SQLiteDatabase ...

  9. Touch事件在移动端web开发中的详解

    一.pc端事件回顾 HTML事件.DOM0事件.DOM2事件 事件对象. 如果上述概念不清楚,请先去了解. 二.移动端事件简介 2.1 pc端事件在移动端的问题 ​ 移动设备主要特点是不配备鼠标,键盘 ...

随机推荐

  1. InfluxDB 的UTC时间问题与简单的持续查询语句

    原文:https://blog.csdn.net/Vblegend_2013/article/details/80904275 最近项目中使用了时序数据库InfluxDB 各方性能也是蛮强大的.但是唯 ...

  2. sql server 清理缓存

    -1. 将当前数据库的全部脏页写入磁盘.“脏页”是已输入缓存区高速缓存且已修改但尚未写入磁盘的数据页. --   CHECKPOINT 可创建一个检查点,在该点保证全部脏页都已写入磁盘,从而在以后的恢 ...

  3. C++进阶--公有继承的二元性

    //########################################################################### /* * 公有继承的两元性 * * - 接口 ...

  4. http系列(一)

    一.关于Url URI由URL和URN组成,URI即统一资源标识符,URL即统一资源定位符,URN即统一资源名称. 现在最常用的是URL. 二.http请求/响应报文 请求报文:请求行.请求头部.空行 ...

  5. tomcat操作

    一.启动 D:\tomcat8.5.9\bin\startup   或者  D:\tomcat8.5.9\bin\catalina start 关闭tomcat: D:\tomcat8.5.9\bin ...

  6. [转]截图软件分享 - Snipaste

    http://chromecj.com/software/2018-10/1538.html https://zh.snipaste.com/download.html

  7. Hbase物理模型设计

    Hbase的存储结构 1.Hbase宏观架构 从上图可以看hbase集群由一个master和多个RegionServer组成,右下角是一个RegionServer的内部图. Hbase的服务器角色构成 ...

  8. HDOJ 2008 数值统计

    #include<iostream> using namespace std; int main() { int n; ) { , y = , z = ; double t; ;i < ...

  9. asp.net mvc 5 单元测试小例子

    using System.Collections.Generic; using System.Linq; using Microsoft.VisualStudio.TestTools.UnitTest ...

  10. 在VMware安装Centos7

    1.新建虚拟机==>典型==>稍后安装操作系统==>选择linux==>centos 64位 2.填写虚拟机名字以及安装位置. 3.磁盘容量采用默认即可. 4. 自定义硬件: ...