谈谈你对ThreadLocal的理解

1. 你知道ThreadLocal是什么吗?

简单地说,就是用来隔离数据的。用ThreadLocal来保存的数据,只对当前线程生效,当前线程对该数据做的任何操作,对别的线程是不生效的。举个栗子一看便知:

public class TestThreadLocal {

    private static User user = new User("jerry", "123"));

    public void fun(User user){
        System.out.println(Thread.currentThread().getName() + "开始执行,修改前的username[" + user.getUsername() + "]");
        user.setUsername("tom");
        System.out.println(Thread.currentThread().getName() + "执行完毕,修改后的username[" + user.getUsername() + "]");
    }

    public static void main(String[] args) {
        TestThreadLocal testThreadLocal = new TestThreadLocal();
        new Thread(() -> {
            testThreadLocal.fun(user);
        }, "线程A").start();

        new Thread(() -> {
            try { TimeUnit.SECONDS.sleep(3); } catch (InterruptedException e) { e.printStackTrace(); }
            testThreadLocal.fun(user);
        }, "线程B").start();
    }
}

这段代码很简单,就是有个user,初始的username为jerry,然后线程A将其改名为tom,3秒钟后,线程B再去读取user的username。因为线程A将起改为tom了,所以线程B开始执行时读取到的应该是tom。看执行结果:

执行结果

分析得没错,线程B开始执行时读取到的确实是tom,说明线程A的修改对线程B生效了。假如用ThreadLocal,结果就不是这样了。

怎么用呢?首先将上述代码中的:

private static User user = new User("jerry", "123"));

改成:

private static ThreadLocal<User> user = ThreadLocal.withInitial(() -> new User("jerry", "123"));

然后将testThreadLocal.fun(user);改成testThreadLocal.fun(user.get());,这样就可以了,再次执行,结果如下:

执行结果

2. 说说ThreadLocal的常用方法

  • set(T t):将值绑定到当前线程中。注意是当前线程,比如你在main线程中set值,再另起两个线程去get,是取不到的;

  • get():当前线程从ThreadLocal中取出自己的副本;

假如你定义了一个静态变量:

private static ThreadLocal<User> threadLocal = new ThreadLocal<>();

那么你必须在线程A和线程B中分别进行set,而不能在主线程中进行set,如下:

new Thread(() -> {
    threadLocal.set(new User("jerry", "123"));
    testThreadLocal.fun(threadLocal.get());
}, "线程A").start();

new Thread(() -> {
    threadLocal.set(new User("jerry", "123"));
    try { TimeUnit.SECONDS.sleep(3); } catch (InterruptedException e) { e.printStackTrace(); }
    testThreadLocal.fun(threadLocal.get());
}, "线程B").start();

jdk8提供了新的构造方式,即开篇中用到的那种方式:

private static ThreadLocal<User> user = ThreadLocal.withInitial(() -> new User("jerry", "123"));

这样就不用在每个线程中进行set了,在这里初始化一次,每个线程get时都会取出属于自己的副本。

3. 对ThreadLocal的应用场景有了解吗?

  • 很多框架中用到了ThreadLocal,比如Spring中,用ThreadLocal来保存数据库连接,这样可以保证单个线程的操作使用的是同一个数据库连接;

  • 可以用ThreadLocal来做session、cookie的隔离;

  • 最经典的一个,SimpleDataFormat调用parse格式化时间的时候,parse方法会先调用Calendar.clear(),再调用Calendar.add(),如果一个线程调用了调用完add,在准备继续parse的时候,另一个线程clear掉了,这就出问题了,所以可以用ThreadLocal;

  • 还有一个就是可以用来传参数,比如你多个方法都要对user对象进行操作,并且有些方法是第三方jar的,不能把user当成参数传过去,那么就可以将user装到ThreadLocal中,要用的时候get出来即可。

4. ThreadLocal的底层原理你造不?

来看看ThreadLocal的set和get方法就秒懂了:

public void set(T value) {
     Thread t = Thread.currentThread();
     ThreadLocalMap map = getMap(t);
     if (map != null)
         map.set(this, value);
     else
         createMap(t, value);
}

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();
}

看到这里是不是就一目了然呢,其实ThreadLocal里面有个ThreadLocalMap,把当前线程作为key,就可以拿到这个线程专属的ThreadLocalMap,所以说每个线程都有一个ThreadLocalMap,拿到了这个ThreadLocalMap后(如果拿到的不为空,就用这个,为空就创建一个,保证每个线程只有一个),将当前ThreadLocal绑定的值设置到map中;get的时候就将map的entry.value返回,就是我们存入的对象啦。

5. 那你对ThreadLocalMap了解过吗?

上面说了,每个线程只对应一个ThreadLocalMap,但是一个线程可以有很多个ThreadLocal来保存不同的对象,那么ThreadLocalMap怎么来保存这多个ThreadLocal呢?那就是用你数组!源码中有这么一个数组:

private Entry[] table;

这个就是用来保存ThreadLocal的。怎么判断当前新加入的ThreadLocal放在数组的哪个位置呢?索引怎么计算出来?ThreadLocalMap会利用usafe类计算出一个threadLocalHashCode,然后再根据:

int i = key.threadLocalHashCode & (len-1)

计算出来的i就是存放当前ThreadLocal的索引。

6. 你用ThreadLocal遇到过什么问题吗?

首先来看看ThreadLocalMap中的这段代码:

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

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

ThreadLocalMap中的key,也就是ThreadLocal对象,被设计成弱引用了,所以在外部没有强引用key的时候,key会被垃圾回收清理掉,ThreadLocalMap中就会出现key为null的Entry,永远无法被GC回收,从而造成内存泄漏。为了避免这个问题,用完ThreadLocal最好调用一下remove方法,手动清除掉。

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

推荐阅读更多精彩内容