ThreadLocal原理

ThreadLocal在Android中的使用场景

当在同一个变量上不同线程保存各自不同的值时,可以使用ThreadLocal,在Android中,与线程间消息传递有关的Looper则使用到ThreadLocal,代码如下:

.......

static final   ThreadLocal    sThreadLocal=new  ThreadLocal();

private static void   prepare(boolean   quitAllowed) {

if(sThreadLocal.get() !=null) {

throw new   RuntimeException("Only one Looper may be created per thread");

}

sThreadLocal.set(new  Looper(quitAllowed));

}

.........

可见,每一个调用Looper.prepare()的线程,都在ThreadLocal里保存了一个Looper对象。

从ThreadLocal.set()方法开始探究,set方法如下:

public void  set(T  value) {

Thread t = Thread.currentThread();

ThreadLocalMap map = getMap(t);

if(map !=null)

map.set(this,value);

else
createMap(t,value);

}

首先拿当前线程的对象,然后从当前线程对象拿一个ThreadLocalMap对象,假设第一次使用ThreadLocal,拿不到ThreadLocalMap对象,从而进入createMap(t,value)方法,该方法如下:

void   createMap(Thread t,T  firstValue) {

t.threadLocals =new  ThreadLocalMap(this,firstValue);

}

可见该方法的作用是设置线程的Thread.threadLocals变量为ThreadLocalMap对象,继续进入ThreadLocalMap的源代码,下面只列出关键代码:

static class  Entry   extendsWeakReference {

Object   value;

Entry(ThreadLocal   k,Object   v) {

super(k);

value= v;

}

}

这是ThreadLocalMap内部类,用于保存值,

private   Entry[ ]   table;

这是ThreadLocalMap的变量,保存值的队列,

ThreadLocalMap(ThreadLocal   firstKey,Object   firstValue) {

table=new     Entry[INITIAL_CAPACITY];

int   i = firstKey.threadLocalHashCode & (INITIAL_CAPACITY-1);

table[i] =new   Entry(firstKey,  firstValue);

size=1;

setThreshold(INITIAL_CAPACITY);

}

这是构造方法,利用ThreadLocal产生数组的下标值,并在该数组位置保存对应值,

private   Entry   getEntry(ThreadLocal   key) {

inti = key.threadLocalHashCode & (table.length-1);

Entry e =table[i];

if(e !=null&& e.get() == key)

return   e;

else

return     getEntryAfterMiss(key,i,e);

}

这个方法是重点方法,解释了如何获取值,首先利用ThreadLocal产生数组下标,找到该位置的值,并进行比较,如果key一致则直接返回,也有可能不一致,因为利用数组实现Map会有哈希表的冲突出现,通常有两种方法解决,一是链地址法,另一个是开放地址法,进入getEntryAfterMiss(key,i,e)方法查看:

private   Entry    getEntryAfterMiss(ThreadLocal key, int  i,Entry e) {

Entry[] tab =table;

int   len = tab.length;

while(e !=null) {

ThreadLocal k = e.get();

if(k == key)

return   e;

if(k ==null)

expungeStaleEntry(i);

else

i =nextIndex(i,len);

e = tab[i];

}

return null;

}

该方法有一个循环,循环里面重点是nextIndex(i,len),继续进入

private static int   nextIndex(int  i, int  len) {

return  ((i +1< len) ? i +1:0);

}

可见,ThreadLocalMap使用开放地址法查找元素,整个ThreadLocalMap的作用就是根据ThreadLocal作为键去保存及查找对应值。

思路回到Thread上,

ThreadLocal.ThreadLocalMap    threadLocals=null;

这是Thread类的代码,可知ThreadLocalMap是保存在Thread类里的变量,其实每个Thread在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();

}

这就是先拿到保存在Thread类自己的成员变量ThreadLocalMap,看getMap(t)方法:

ThreadLocalMap   getMap(Thread   t) {

returnt.threadLocals;

}

是获取当前线程的变量ThreadLocalMap变量,然后根据ThreadLocalMap以ThreadLocal作为key查找值。

因此ThreadLocal的原理就是,我们可以new 很多ThreadLocal变量,每个线程保存一个不同的值,最终这个值以ThreadLocal为key保存在不同的线程的ThreadLocalMap变量上,当要拿出值时,每个线程最终也是从自己的ThreadLocalMap变量上拿值,所以最终效果就是在同一个ThreadLocal变量上,不同线程可以保存获取不同的值

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

推荐阅读更多精彩内容