日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

对ThreadLocal实现原理的一点思考

發(fā)布時(shí)間:2025/3/21 编程问答 24 豆豆
生活随笔 收集整理的這篇文章主要介紹了 对ThreadLocal实现原理的一点思考 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

前言


在《透徹理解Spring事務(wù)設(shè)計(jì)思想之手寫實(shí)現(xiàn)》中,已經(jīng)向大家揭示了Spring就是利用ThreadLocal來實(shí)現(xiàn)一個(gè)線程中的Connection是同一個(gè),從而保證了事務(wù)。本篇博客將帶大家來深入分析ThreadLocal的實(shí)現(xiàn)原理。

?

ThreadLocal是什么、有什么、能做什么?

ThreadLocal提供一個(gè)線程(Thread)局部變量,訪問到某個(gè)變量的每一個(gè)線程都擁有自己的局部變量。說白了,ThreadLocal就是想在多線程環(huán)境下去保證成員變量的安全。

ThreadLocal提供的方法

ThreadLocal API

對(duì)于ThreadLocal而言,常用的方法,就是get/set/initialValue方法。

我們先來看一個(gè)例子

demo

運(yùn)行結(jié)果

是你想象中的結(jié)果么?

?

很顯然,在這里,并沒有通過ThreadLocal達(dá)到線程隔離的機(jī)制,可是ThreadLocal不是保證線程安全的么?這是什么鬼?

雖然,ThreadLocal讓訪問某個(gè)變量的線程都擁有自己的局部變量,但是如果這個(gè)局部變量都指向同一個(gè)對(duì)象呢?這個(gè)時(shí)候ThreadLocal就失效了。仔細(xì)觀察下圖中的代碼,你會(huì)發(fā)現(xiàn),threadLocal在初始化時(shí)返回的都是同一個(gè)對(duì)象a!

?

看一看ThreadLocal源碼

我們直接看最常用的set操作:

set

?

線程局部變量

?

createMap

你會(huì)看到,set需要首先獲得當(dāng)前線程對(duì)象Thread;

然后取出當(dāng)前線程對(duì)象的成員變量ThreadLocalMap;

如果ThreadLocalMap存在,那么進(jìn)行KEY/VALUE設(shè)置,KEY就是ThreadLocal;

如果ThreadLocalMap沒有,那么創(chuàng)建一個(gè);

說白了,當(dāng)前線程中存在一個(gè)Map變量,KEY是ThreadLocal,VALUE是你設(shè)置的值。

看一下get操作:

get

這里其實(shí)揭示了ThreadLocalMap里面的數(shù)據(jù)存儲(chǔ)結(jié)構(gòu),從上面的代碼來看,ThreadLocalMap中存放的就是Entry,Entry的KEY就是ThreadLocal,VALUE就是值。

ThreadLocalMap.Entry:

弱引用?

在JAVA里面,存在強(qiáng)引用、弱引用、軟引用、虛引用。這里主要談一下強(qiáng)引用和弱引用。

強(qiáng)引用,就不必說了,類似于:

A a = new A();

B b = new B();

考慮這樣的情況:

C c = new C(b);

b = null;

考慮下GC的情況。要知道b被置為null,那么是否意味著一段時(shí)間后GC工作可以回收b所分配的內(nèi)存空間呢?答案是否定的,因?yàn)榧幢鉨被置為null,但是c仍然持有對(duì)b的引用,而且還是強(qiáng)引用,所以GC不會(huì)回收b原先所分配的空間!既不能回收利用,又不能使用,這就造成了內(nèi)存泄露

那么如何處理呢?

可以c = null;也可以使用弱引用!(WeakReference w = new WeakReference(b);)

分析到這里,我們可以得到:

內(nèi)存結(jié)構(gòu)圖

這里我們思考一個(gè)問題:ThreadLocal使用到了弱引用,是否意味著不會(huì)存在內(nèi)存泄露呢?

首先來說,如果把ThreadLocal置為null,那么意味著Heap中的ThreadLocal實(shí)例不再有強(qiáng)引用指向,只有弱引用存在,因此GC是可以回收這部分空間的,也就是key是可以回收的。但是value卻存在一條從Current Thread過來的強(qiáng)引用鏈。因此只有當(dāng)Current Thread銷毀時(shí),value才能得到釋放。

因此,只要這個(gè)線程對(duì)象被gc回收,就不會(huì)出現(xiàn)內(nèi)存泄露,但在threadLocal設(shè)為null和線程結(jié)束這段時(shí)間內(nèi)不會(huì)被回收的,就發(fā)生了我們認(rèn)為的內(nèi)存泄露。最要命的是線程對(duì)象不被回收的情況,比如使用線程池的時(shí)候,線程結(jié)束是不會(huì)銷毀的,再次使用的,就可能出現(xiàn)內(nèi)存泄露。

那么如何有效的避免呢?

事實(shí)上,在ThreadLocalMap中的set/getEntry方法中,會(huì)對(duì)key為null(也即是ThreadLocal為null)進(jìn)行判斷,如果為null的話,那么是會(huì)對(duì)value置為null的。我們也可以通過調(diào)用ThreadLocal的remove方法進(jìn)行釋放!

?

好了,到這里,ThreadLocal的剖析就完成了,自己對(duì)ThreadLocal的認(rèn)識(shí)又深入了些,^_^

?

手寫系列相關(guān)爆文


【手寫系列】寫出我的第一個(gè)框架:迷你版Spring MVC

【手寫系列】透徹理解Spring事務(wù)設(shè)計(jì)思想之手寫實(shí)現(xiàn)

【手寫系列】透徹理解MyBatis設(shè)計(jì)思想之手寫實(shí)現(xiàn)

【手寫系列】純手寫實(shí)現(xiàn)一個(gè)高可用的RPC

【手寫系列】理解數(shù)據(jù)庫(kù)連接池底層原理之手寫實(shí)現(xiàn)

【手寫系列】對(duì)HashMap的思考及手寫實(shí)現(xiàn)

【手寫系列】純手寫實(shí)現(xiàn)JDK動(dòng)態(tài)代理

【手寫系列】寫一個(gè)迷你版的Tomcat



作者:張豐哲
鏈接:https://www.jianshu.com/p/ee8c9dccc953
來源:簡(jiǎn)書
簡(jiǎn)書著作權(quán)歸作者所有,任何形式的轉(zhuǎn)載都請(qǐng)聯(lián)系作者獲得授權(quán)并注明出處。

總結(jié)

以上是生活随笔為你收集整理的对ThreadLocal实现原理的一点思考的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。