ThreadLocal和InheritableThreadLocal

ThreadLocal

概述

ThreadLocal一般称为线程本地变量,它是一种特殊的线程绑定机制,将变量与线程绑定在一起,为每一个线程维护一个独立的变量副本,每个线程的私有变量。通过ThreadLocal可以将对象的可见范围限制在同一个线程内

一个误区

不要将synchronized与ThreadLocal进行对比,sysnchronized是一种互斥同步机制,是为了保证在多线程环境下对于共享资源的正确访问。而ThreadLocal是提供一个“线程级”的变量作用域(线程内的static变量),它是一种线程封闭(每个线程独享变量)技术,更直白点讲,ThreadLocal可以理解为将对象的作用范围限制在一个线程上下文中,使得变量的作用域为“线程级”。

再简单点总结,Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。

没有ThreadLocal的时候,一个线程在其生命周期内,可能穿过多个层级,多个方法,如果有个对象需要在此线程周期内多次调用,且是跨层级的(线程内共享),通常的做法是通过参数进行传递;而ThreadLocal将变量绑定在线程上,在一个线程周期内,无论“你身处何地”,只需通过其提供的get方法就可轻松获取到对象。极大地提高了对于“线程级变量”的访问便利性。

用法:

一般把ThreadLocal变量作为单独的public static变量

通过set(...)存入数据,通过get()获取数据

解决get返回null问题

第一次使用get都会返回null,可以继承Threadlocal再覆盖initialValue方法可以解决

private static final ThreadLocal<Integer> threadId = 
    new ThreadLocal<Integer>(){
       @Override
       protected Integer initialValue(){
           return 1;
       }
    }

看看set源码

public void set(T value) {
    Thread t = Thread.currentThread()   //首先获取当前线程对象
    ThreadLocalMap map = getMap(t);     //获取该线程对象的ThreadLocalMap
    if (map != null)
        map.set(this, value);
        //如果map不为空,执行set操作,以当前threadLocal对象为key,实际存储对象为value进行set操作
    else
        createMap(t, value);            //如果map为空,则为该线程创建ThreadLocalMap
}

ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;              //线程对象持有ThreadLocalMap的引用
}

void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

从上面代码可以看出来,ThreadLocal是通过在Thread里面保存了一个ThreadLocalMap才产生了全局线程私有变量,在Thread中,也确实有这么一行

ThreadLocal.ThreadLocalMap threadLocals = null;

这样子就大概知道ThreadLocal的设计思路了:

  1. ThreadLocal作为变量访问的入口
  2. 每个Thread含有一个ThreadLocalMap对象,这个ThreadLocalMap持有对象的引用
  3. ThreadLocalMap以当前的threadlocal对象为key,以真正的存储对象为value。get时通过threadlocal实例就可以找到绑定在当前线程上的对象。

看上去似乎变为Map<Thread,T>这种形式会自然些,一个线程对应一个存储对象。ThreadLocal这样设计的目的主要有两个:

  1. 保证当前线程结束时相关对象能尽快被回收,把value放在了线程当中,这样线程死亡,value随之回收
  2. ThreadLocalMap中的元素会大大减少,而map过大更容易造成哈希冲突而导致性能变差

get源码

public T get() {
    Thread t = Thread.currentThread();  //首先获取当前线程
    ThreadLocalMap map = getMap(t);     //获取线程的map对象
    if (map != null) {      //如果map不为空,以threadlocal实例为key获取到对应Entry,然后从Entry中取出对象即可。
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null)
            return (T)e.value;
    }
    return setInitialValue();
    //如果map为空,也就是第一次没有调用set直接get(或者调用过set,又调用了remove)时,为其设定初始值
}

private T setInitialValue() {
    T value = initialValue();
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
    return value;
}

关于ThreadLocal的使用问题

如果我们为一个线程关联的对象是“完全独享”的,也就是每个线程拥有一整套的新的栈中的对象引用+堆中的对象,那么这种情况下是真正的线程独享变量”,相当于一种深度拷贝,每个线程自己玩自己的,对该对象做任何的操作也不会对别的线程有任何影响。(对于对象过大的时候,如果每个线程有一个深拷贝,开销很大)

另一种更普遍的情况,所谓的独享变量副本,其实也就是每个线程都拥有一个独立的对象引用,而堆中的对象还是线程间共享的,这种情况下,自然还是会涉及到对共享资源的访问操作,依然会有线程不安全的风险。所以说,ThreadLocal无法解决线程安全问题。

为何threadLocals的类型ThreadLocalMap的键值为ThreadLocal对象,因为每个线程中可有多个threadLocal变量

另外还有一个坑

当使用不当时,其作用域范围使用者可能不明确。

解释一下:线程如果运行完,也就是run方法执行完了,线程注销,ThreadLocal被回收;可是,还有一种更加常见的情况,线程池!线程可能不会立马注销,也就是ThreadLocal不会被回收,如果ThreadLocal中包装了集合类或复杂对象,那内部集合类和复杂对象锁占用的内存可能会膨胀

使用ThreadLocal的场景

最常见的ThreadLocal使用场景为 用来解决 数据库连接、Session管理等。

以Spring为例,Spring的事务管理器通过AOP切入业务代码,在进入业务代码前,会依据相应的事务管理器提取出相应的事务对象,假如事务管理器是DataSourceTransactionManager,就会从DataSource中获取一个连接对象,通过一定的包装后将其保存在ThreadLocal中。而且Spring也将DataSource进行了包装,重写了当中的getConnection()方法,或者说该方法的返回将由Spring来控制,这样Spring就能让线程内多次获取到的Connection对象是同一个。通过将连接对象保存在线程内部,不论什么时候都能拿到,此时Spring很清楚什么时候回收这个连接,也就是很清楚什么时候从ThreadLocal中删除这个元素

再来看看InheritableThreadLocal

作用:

显然,在原本的ThreadLocal使用上,如果父线程创建了子线程,父线程是没办法把当前拥有的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)
            return (T)e.value;
    }
    return setInitialValue();
}

第一步获取当前线程,而当前线程为子线程,子线程有自己的ThreadLocalMap,跟父线程毫无关系,这个地方可以从2点佐证:1. ThreadLocal的目的是为了使得每个线程有自己的独立私有变量

  1. Thread的创建会调用init(...),当中不会为子线程继承ThreadLocal(这里详细可见Thread的init()函数)

有些时候需要使得父线程的一些变量继承给子线程,这时候就出现了InheritableThreadLocal

原理

用法跟ThreadLocal一样,所以就只讲讲InheritableThreadLocal的实现,直接上源码

public class InheritableThreadLocal<T> extends ThreadLocal<T> {
    protected T childValue(T parentValue) {
        return parentValue;
    }
    ThreadLocalMap getMap(Thread t) {
       return t.inheritableThreadLocals;
    }
    void createMap(Thread t, T firstValue) {
        t.inheritableThreadLocals = new ThreadLocalMap(this, firstValue);
    }
}

看样子,这里的关键在于getMap这个地方,由于是继承,所以会使用覆盖的方法,原本的getMap(Thread t)是长这样的

ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;  
}

只是返回的map不同,InheritableThreadLocal返回的map是从父线程继承下来的,但这是在什么时候设置的?最重点的地方到了,就是在每个Thread创建的时候都会调用的init()方法,这里只贴出跟

private void init(ThreadGroup g, Runnable target, 
            String name, long stackSize, AccessControlContext acc) {
    ......    
    if (inheritThreadLocals && parent.inheritableThreadLocals != null)        
        this.inheritableThreadLocals = 
            ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);
    ......    
}

来到了ThreadLocal的方法

static ThreadLocalMap createInheritedMap(ThreadLocalMap parentMap) {
    return new ThreadLocalMap(parentMap);       //这里把parentMap给复制了一遍
}

到此

参考:

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

推荐阅读更多精彩内容