ThreadLocal初步探秘

ThreadLocal是什么?

ThreadLocal是java语言实现的一种线程本地存储的方式,有时候我们会遇到这样一种需求:每条线程都要去存取一个同名的变量,但是每条线程中该变量的值都是不一样的,那么如果让我们去实现的话,会采用什么方式去实现呢? 也许你会认为你可以去使用一个线程共享的Map<Thread,Object>, 其中Map中的key为线程对象而value则是我们需要存储的值,然后通过map.get(Thread.currentThread())来获取本线程中该变量的值.

如果不考虑同步和效率的问题的话,这种实现方式是完全可以的,但是,问题的关键在于,如果有很多线程进行操作这个Map呢?为了保证线程的安全性,势必要对Map进行加锁,每当有一个线程在操作这个map时,其他线程只能去等待锁资源的释放,这种性能是我们不能够容忍的。

也许你会反驳说我们可以使用java.util.concurrenct.*包下面的ConcurrencyHashMap来提高并发率,但是它只是降低了锁的粒度,并没有从根本上避免同步锁,而jdk提供的ThreadLocal则很好的解决了这个问题。

我们首先先来看一下ThreadLocal的简单用法

//为不同的线程关联不同的用户ID
public class ThreadLocalDemo {

    private static volatile ThreadLocal<String> userID = new ThreadLocal<>();

    public static void main(String[] args) {

        Runnable r = new Runnable() {
            @Override
            public void run() {
                String name = Thread.currentThread().getName();
                if (name.equals("A")){
                    userID.set("aaaaaaa");
                }else if (name.equals("B")){
                    userID.set("bbbbbbb");
                }


                System.out.println("ThreadName:"+name+" userID: "+userID.get());

            }
        };

        Thread t1 = new Thread(r);
        Thread t2 = new Thread(r);
        t1.setName("A");
        t2.setName("B");
        t1.start();
        t2.start();
    }

}

输出结果

ThreadName:A userID: aaaaaaa
ThreadName:B userID: bbbbbbb

从这个例子中我们可以看出,虽然说是共享了同一个ThreadLocal,但是对于每一个线程来说,它们可以存储自己的值,不同线程存储的内容互不相干。

看完了ThreadLocal的简单用法之后,让我们再去探索一下ThreadLocal的使用原理

我使用的是jdk1.8,通过分析源码我们可以发现,不同线程和本地存储的变量值的映射关系是由一个称之为ThreadLoaclMap的类来实现的,从名字中我们可以看出,它是一个Map,对于Map来说,就应该有键值对,现在我们已经知道值是我们存储的变量值了,那么键是什么呢?这可能跟我们最初的想法不一样,它的键并不是我们原以为的Thread对象,而是一个ThreadLocal。那你可能会产生疑问,那它又是怎样实现Thread和value的关联的呢?

在这里它采用的方式是由Thread类来管理ThreadLocalMap对象.

打开Thread.class我们可以看到, Thread类持有一个ThreadLocalMap
threadlocalmap.png

从类型的定义来看,我们可以看出ThreadLocalMap是ThreadLocal的内部类,那么这个内部类又是怎么定义的呢?让我们进入ThreadLocal的源码实现去看一下

 static class ThreadLocalMap {
  
        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }
      private Entry[] table;

    //.......省略以下代码

 }

从这里我们可以看出来,ThreadLocalMap是由一个Entry数组table构成的,也就是说,是由一个一个的Entry构成的,我们还可以发现,Entry继承了 WeakReference<ThreadLocal<?>,在构造Entry的时候也调用了super(k),这里为什么会牵扯到弱引用呢?这跟使用ThreadLocal可能会造成的内存泄露的风险有关系,如果key是弱引用,当没有指向key的强引用后,在进行垃圾回收时,就会把这个key回收掉,至于具体的原因,我们会到后面进行分析。

大体结构我们应该清楚了,也就是说,ThreadLocalMap实际存储本地值,ThreadLocalMap是ThreadLocal的一个静态内部类,ThreadLocal实例的set(),get()等方法是对ThreadLocalMap的实际操作来设置线程本地存储的值和获取该值,而ThreadLocalMap又是由Thread来管理的。。。到了这里你如果没有被绕晕,那么恭喜你到目前为止的内容你大概都理解了,如果被绕晕的话,也没关系,我们还有下面这一张图。

ThreadLocal.jpg

这样一来,Thread,ThreadLocalMap,ThreadLocal这三者的关系我们已经清楚了,那么ThreadLocal是怎样具体实现对本地存储值的set()和get()呢?

继续从源码开始探究。。

public void set(T value) {
    //当执行set方法的时候,首先会获取当前线程的对象
    Thread t = Thread.currentThread();
    //根据线程对象获取指定线程里面持有的ThreadLocalMap
    ThreadLocalMap map = getMap(t);
    //如果我们获取到了map,这个map存在不为空
    if (map != null)
        //对map进行set操作,key为调用该set方法的ThreadLocal对象,值为我们需要存储的值
        map.set(this, value);
    else
        //如果map为空,则调用createMap()来创建一个
        createMap(t, value);
}

public T get() {
    //首先获得当前线程
    Thread t = Thread.currentThread();
     //根据线程对象获取指定线程里面持有的ThreadLocalMap
    ThreadLocalMap map = getMap(t);
      //如果我们获取到了map,这个map存在不为空
    if (map != null) {
        //以调用该方法的ThreadLocal对象,也就是this为key,获取Entry
        ThreadLocalMap.Entry e = map.getEntry(this);
        //如果获取的Entry不为空
        if (e != null) {
            @SuppressWarnings("unchecked")
            //获取Entry的value值返回
            T result = (T)e.value;
            return result;
        }
    }
    //如果为空,调用setInitialValue()并返回
    return setInitialValue()并返回
}

//根据线程对象获取指定线程里面持有的ThreadLocalMap
ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}


由于每一条线程均含有各自私有的ThreadLocalMap容器,这些容器相互独立互不影响,因此不会存在线程安全性问题,从而也无需使用同步机制来保证多条线程访问容器的互斥性。

关注一下ThreadLocal内存泄露的问题

在ThreadLocalMap中,只有key是弱引用,value仍然是一个强引用。当某一条线程中的ThreadLocal使用完毕,没有强引用指向它的时候,这个key指向的对象就会被垃圾收集器回收,从而这个key就变成了null;然而,此时value和value指向的对象之间仍然是强引用关系,只要这种关系不解除,value指向的对象永远不会被垃圾收集器回收,从而导致内存泄漏!

不过不用担心,ThreadLocal提供了这个问题的解决方案。

每次操作set、get、remove操作时,ThreadLocal都会将key为null的Entry删除,从而避免内存泄漏。

那么问题又来了,如果一个线程运行周期较长,而且将一个大对象放入LocalThreadMap后便不再调用set、get、remove方法,此时该仍然可能会导致内存泄漏。

这个问题确实存在,没办法通过ThreadLocal解决,而是需要程序员在完成ThreadLocal的使用后要养成手动调用remove的习惯,从而避免内存泄漏。

ThreadLocal的 应用场景

1.Web系统Session的存储就是ThreadLocal一个典型的应用场景。

Web容器采用线程隔离的多线程模型,也就是每一个请求都会对应一条线程,线程之间相互隔离,没有共享数据。这样能够简化编程模型,程序员可以用单线程的思维开发这种多线程应用。

当请求到来时,可以将当前Session信息存储在ThreadLocal中,在请求处理过程中可以随时使用Session信息,每个请求之间的Session信息互不影响。当请求处理完成后通过remove方法将当前Session信息清除即可。

2.Spring的事务管理器通过AOP切入业务代码,在进入业务代码前,会根据对应的事务管理器提取出相应的事务对象,为了获得同一个Connection,Spring在这里也巧妙利用了ThreadLocal的特性

假如事务管理器是DataSourceTransactionManager,就会从DataSource中获取一个连接对象,通过一定的包装后将其保存在ThreadLocal中。并且Spring也将DataSource进行了包装,重写了其中的getConnection()方法,或者说该方法的返回将由Spring来控制,这样Spring就能让线程内多次获取到的Connection对象是同一个。

为什么要放在ThreadLocal里面呢?因为Spring在AOP后并不能向应用程序传递参数,应用程序的每个业务代码是事先定义好的,Spring并不会要求在业务代码的入口参数中必须编写Connection的入口参数。此时Spring选择了ThreadLocal,通过它保证连接对象始终在线程内部,任何时候都能拿到,此时Spring非常清楚什么时候回收这个连接,也就是非常清楚什么时候从ThreadLocal中删除这个元素

该篇博客是我的一些总结,很多内容是参考了大闲人柴毛毛的博客总结,加上我的一些个人总结,为了尊重原作,

我附上了他的博客地址https://juejin.im/post/5aa74967f265da23a334e373

作者:lhsjohn

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容

  • 下面我就以面试问答的形式学习我们的——ThreadLocal类(源码分析基于JDK8) 问答内容 1、问:Thre...
    Sophia_dd35阅读 2,078评论 1 36
  • 移步Android Handler机制详解[https://www.jianshu.com/p/e37e2db2b...
    凯玲之恋阅读 807评论 0 0
  • 原理 产生线程安全问题的根源在于多线程之间的数据共享。如果没有数据共享,就没有多线程并发安全问题。ThreadLo...
    Java耕耘者阅读 300评论 0 0
  • 1 老人其实很有意思,一边盼着儿女有所成就,满足自己的愿望,可你要是给她买个啥东西,她就嫌你浪费,就说钱要用在刀刃...
    非凡说阅读 204评论 0 1
  • 突然心静 很多事情没有想象的那么难嘛 生活还是有很多妙的不得了东西 怎么会有那么美丽的诗 所以 没事就要多笑笑 温...
    二人余安阅读 383评论 0 3