ThreadLocal

[toc]

设计理念

为每个线程创造一个资源的复本。将每一个线程存取数据的行为加以隔离,实现的方法就是给予每个线程一个特定空间来保管该线程所独享的资源。

作用

为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副本,而不会和其它线程的副本冲突。从线程的角度看,就好像每一个线程都完全拥有该变量。

使用

ThreadLocal实例通常是类中的private static字段,它们希望将状态与某一个线程相关联。每个线程都保持对其线程局部变量副本的隐式引用,只要线程是活动的并且ThreadLocal实例是可访问的,在线程消失之后,其线程局部实例的所有副本都会被垃圾回收(除非存在对这些副本的其他引用)。
首先创建ThreadLocal对象:

ThreadLocal<Integer> mValue = new ThreadLocal<Integer>();

然后在线程中调用set和get方法来设置和获取值,例如:

mValue.set(1); 
int value=mValue.get();

实现原理

简单地说,就是在ThreadLocal类中有一个静态Map,用于存储每一个线程的变量的副本。Map的Key是Thread,value就是副本的值。

深入源码去看,ThreadLocal把线程和线程局部变量存在ThreadLocalMap中,而ThreadLocalMap是ThreadLocal的静态类部类,我们来看看ThreadLocalMap的部分源码:

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

这个Map的key是ThreadLocal对象的弱引用,当要抛弃掉ThreadLocal对象时,垃圾收集器会忽略这个key的引用而清理掉ThreadLocal对象 。

那么到底是ThreadLocal还是Thread持有ThreadLocalMap对象的引用呢?
我们在Thread源码中发现:

/* ThreadLocal values pertaining to this thread.  
 * This map is maintained by the ThreadLocal class. 
 */ 
ThreadLocal.ThreadLocalMap threadLocals = null;

ThreadLocalMap变量属于Thread的内部属性,不同的Thread拥有完全不同的ThreadLocalMap变量.    
Thread中的ThreadLocalMap变量的值是在ThreadLocal对象进行set或者get操作时创建的.   
在创建ThreadLocalMap之前,会首先检查当前Thread中的ThreadLocalMap变量是否已经存在,如果不存在则创建一个;如果已经存在,则使用当前Thread已创建的ThreadLocalMap.

ThreadLocal的接口方法

/** 
* 返回此线程局部变量的初始值 
* 
* 线程第一次使用 get() 方法访问变量时将调用此方法,但如果线程之前调用 
* 了 set(T) 方法,则不会对该线程再调用 initialValue 方法。通常,此 
* 方法对每个线程最多调用一次,但如果在调用 get() 后又调用了  
* remove(),则可能再次调用此方法。 
*  
* 该实现返回 null;如果程序员希望线程局部变量具有 null 以外的值,则 
* 必须为 ThreadLocal 创建子类,并重写此方法。通常将使用匿名内部类完 
* 成此操作。 
*/ 
protected T initialValue(); 
 
/** 
* 返回此线程局部变量的当前线程的值 
*  
* 如果变量没有用于当前线程的值,则先 
* 将其初始化为调用initialValue() 方法返回的值。 
*/ 
public T get(); 
 
/** 
* 将此线程局部变量的当前线程副本中的值设置为指定值 
*  
* 大部分子类不需要重写此方法,它们只依靠 initialValue() 方法 
* 来设置线程局部变量的值 
*/ 
public void set(T value); 
 
/** 
* 移除此线程局部变量当前线程的值 
*  
* 如果此线程局部变量随后被当前线程 读取,且这期间当前线程没有 
* 设置其值,则将调用其 initialValue() 方法重新初始化其值。 
* 这将导致在当前线程多次调用 initialValue 方法。 
*/ 
public void remove();

如果希望线程局部变量初始化其它值,那么需要自己实现ThreadLocal的子类并重写该方法,通常使用一个内部类对ThreadLocal进行实例化。
比如下面的例子,SerialNum类为每一个类分配一个序号:

public class SerialNum { 
    // The next serial number to be assigned 
    private static int nextSerialNum = 0; 
 
    private static ThreadLocal serialNum = new ThreadLocal() { 
        protected synchronized Object initialValue() { 
            return new Integer(nextSerialNum++); 
        } 
    } 
 
    public static int get() { 
        return ((Integer) (serialNum.get())).intValue(); 
    } 
}

ThreadLocal如何做到线程安全

从上面的分析我们可以得出:
• 因为每个Thread在进行对象访问时,访问的都是各自线程自己的ThreadLocalMap,所以保证了Thread与Thread之间的数据访问隔离。
• 不同的ThreadLocal实例操作同一Thread时,ThreadLocalMap在存储时采用当前ThreadLocal的实例作为key来保证数据访问隔离(上面源码Entry处可以看出)。

举个例子说明:

1.  public class Test{ 
2.      static ThreadLocal<Integer> local = new ThreadLocal<Integer>(){ 
3.          protected synchronized Integer initialValue(){   
4.              return 0;   
5.          } 
6.      };   
7.   
8.      public static void main( String args[]){ 
9.           testRun t = new testRun(); 
10.          new Thread(t).start(); 
11.          new Thread(t).start(); 
12.     } 
13.  
14.     public static  class testRun implements Runnable{ 
15.     @Override 
16.     public void run() { 
17.         // TODO Auto-generated method stub 
18.         for(int i=5 ;i>0;i--){ 
19.             try {   
20.                 Thread.sleep(1000);   
21.             } catch (InterruptedException e) {   
22.                 e.printStackTrace();   
23.             } 
24.             System.out.println(Thread.currentThread().getName()+"::"+local.get()); 
25.  
26.              int localValue=local.get();   
27.              local.set(++localValue);   
28.                 }  
29.         } 
30.     }    
31.    }

输出结果为:

1.  Thread-0::0 
2.  Thread-1::0 
3.  Thread-0::1 
4.  Thread-1::1 
5.  Thread-1::2 
6.  Thread-0::2 
7.  Thread-1::3 
8.  Thread-0::3 
9.  Thread-0::4 
10. Thread-1::4

TheadLocal模式与同步机制的区别

1.实现机制

同步机制采用了“以时间换空间”的方式,提供一份变量,让不同的线程排队访问.
而ThreadLocal采用了“以空间换时间”的方式,为每一个线程都提供一份变量的副本,从而实现同时访问而互不影响。

2.同步共享方面

Java中的synchronized是一个保留字,它依靠JVM的锁机制来实现临界区的函数或者变量的访问中的原子性.在同步机制中,通过对象的锁机制保证同一时间只有一个线程访问变量.此时,被用作“锁机制”的变量是多个线程共享的;

而ThreadLocal会为每一个线程维护一个和该线程绑定的变量的副本,从而隔离了多个线程的数据,每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。

3.使用场合

同步机制是为了同步多个线程对相同资源的并发访问,是为了多个线程之间进行通信的有效方式。
而ThreadLocal是隔离多个线程的数据共享,从根本上就不在多个线程之间共享资源(变量),这样当然不需要对多个线程进行同步了。
所以,如果你需要进行多个线程之间进行通信,则使用同步机制。如果需要隔离多个线程之间的共享冲突,可以使用ThreadLocal。

https://www.jianshu.com/p/069d0d8b5427

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