美文网首页
ThreadLocal庐山真面目

ThreadLocal庐山真面目

作者: 来个Android小哥 | 来源:发表于2020-05-26 13:23 被阅读0次

    目录: 

              1.什么是ThreadLocal

              2.ThreadLocal的使用

              3.ThreadLocal原理

              5.面试与总结

      1.什么是ThreadLocal?

              ThreadLocal被称为线程本地变量,有些其它地方叫做线程本地存储,ThreadLocal可以让每个线程拥有一个属于自己数据副本,并且不会和其它线程的数据副本产生冲突,这也就是我们经常听到的线程数据隔离。它最主要的目的是防止线程操作同一个变量时,产生冲突。

    2.ThreadLocal的使用

            其实我们平常想要实现数据有一个隔离作用的话会想到用map,它的实现就是key-value的机制,它可以让我们实现和ThreadLocal相同的作用,那么我们线程为什么不用map,而用ThreadLocal呢?所以有了以下分析,首先我们先手写一个用map和ThreadLocal存取数据的案例。

    map存取数据:

    我们自定义了一个MyThreadLocal类,里面有一个容器map,它是以当前线程为key,以传进来的值为value,进行存进操作,然后再创建3个线程,并用我们实例化的MyThreadLocal存值,最后打印结果为:

    看这个打印结果是我们想要的,一一对应了。然后我们用jdk提供的ThreadLocal 进行操作如下:其它代码不变,

    直接实例化jdk中的ThreadLocal,继续打印结果:

    从这里可以看出,实现的效果是一样的(也可以说结果是完全一样的),那为什么jdk不直接用map,而是采用新增了一个Threadlocal进行数据隔离呢?

    使用map会出现的问题:我们思考一下,假如多个线程操作一个大map,那么多个线程去抢有map的存取权,它们之间肯定存在一个激烈的竞争。虽然我们对外面(用户)暴露的确实每个线程是有一个自己的数据副本,但是在我们的内部实现上会产生一个激烈的竞争。在这里画个图理解一下:

    在这里解释一下,我们把上述圆环比做一个一个人,大矩形比做一个盛碗的大箩筐,大箩筐里是一个个小碗,当一群人去吃饭时,都去激烈的竞争一个大箩筐使用权,因为大箩筐有属于自己的吃饭工具(碗),所以这样很不友好,就必须改变这种现象......

    3.ThreadLocal原理:

       原理就必须看源码了,首先,从先前例子上看ThreadLocal 有一个set() 和get() 方法:

    方法作用就是设置值和取出值,我们从set开始看,点进去,有如下代码:

    (不管是set() 操作还是get() 操作都是在ThreadLocal类里进行操作的)

    从这里我们可以看到,第200行代码,是获取当前线程的实例,从哪个线程进行set操作,就是获取哪个线程的实例,接着看201行会调用一个getMap(t),传进去一个线程t,我们进入getMa p()方法看有如下代码:

    在这里只有两行代码,其实很简单,就是获取当前线程的一个ThreadLocalMap类,

    接下来我们找到ThreadLocal中的 ThreadLocalMap类中:

    我们看划红线的地方,可以看到ThreadLocalMap中放了一个Entry[] 数组,在Entry类中,我们可以看到,Entry的构造方法,有一个以ThreadLocal为key,object为value的一个数据,至此,我们理一下流程:

    它是一层包着一层的:

    我们看get()方法

    第161行当然是先获取当前线程的ThreadLocalMap 类,在第163行可以看到,是以当前ThreadLocal 的实例为key(因为不管是set方法还是get方法都是在ThreadLocal方法中进行的,this就是代表当前ThreadLocal对象),去获取一个Entry,最后再获取值的。进入getEntry中看:

    可以看到先计算一下下标,随后再从数组中根据下标提取一个Entry值。再返回出去。

    至此,我们的ThreadLocal原理分析的差不多了,可能会遗漏些或者哪里可能表达的不够清楚,希望看官们多多提意见。

     4.面试与总结

           目前看,基本上谈到多线程的问题,一般都会问到ThreadLocal 的原理,ThreadLocal处理的实在是很巧妙,我们可以直接这样理解,它内部直接将一个ThreadLocalMap挂到了当前的线程中,而这个map只属于当前的线程,所以很好的实现了线程隔离。

         很多时候,看源码真的会提高自己的水平,因为源码的思维总是会给你眼前一亮,不要害怕看源码,因为看多了,自然就懂的怎么去看了,有时候还得借鉴源码的实现手法,这也是提高自己的一种手段。这就是源码之美吧。

    相关文章

      网友评论

          本文标题:ThreadLocal庐山真面目

          本文链接:https://www.haomeiwen.com/subject/sdoeahtx.html