亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

淺談Java引用和Threadlocal的那些事

2024-07-14 08:43:58
字體:
來源:轉載
供稿:網友

1 背景

某一天在某一個群里面的某個群友突然提出了一個問題:"threadlocal的key是虛引用,那么在threadlocal.get()的時候,發生GC之后,key是否是null?"屏幕前的你可以好好的想想這個問題,在這里我先賣個關子,先講講Java中引用和ThreadLocal的那些事。

2 Java中的引用

對于很多Java初學者來說,會把引用和對象給搞混淆。下面有一段代碼,

User zhangsan = new User("zhangsan", 24);

這里先提個問題zhangsan到底是引用還是對象呢?很多人會認為zhangsan是個對象,如果你也是這樣認為的話那么再看一下下面一段代碼

User zhangsan;zhangsan = new User("zhangsan", 24);

這段代碼和開始的代碼其實執行效果是一致的,這段代碼的第一行User zhangsan,定義了zhangsan,那你認為zhangsan還是對象嗎?如果你還認為的話,那么這個對象應該是什么呢?的確,zhangsan其實只是一個引用,對JVM內存劃分熟悉的同學應該熟悉下面的圖片:

Java,引用,Threadlocal

 其實zhangsan是棧中分配的一個引用,而new User("zhangsan", 24)是在堆中分配的一個對象。而'='的作用是用來將引用指向堆中的對象的。就像你叫張三但張三是個名字而已并不是一個實際的人,他只是指向的你。

我們一般所說的引用其實都是代指的強引用,在JDK1.2之后引用不止這一種,一般來說分為四種:強引用,軟引用,弱引用,虛引用。而接下來我會一一介紹這四種引用。

2.1 強引用

上面我們說過了 User zhangsan = new User("zhangsan", 24);這種就是強引用,有點類似C的指針。對強引用他的特點有下面幾個:

強引用可以直接訪問目標對象。

只要這個對象被強引用所關聯,那么垃圾回收器都不會回收,那怕是拋出OOM異常。

容易導致內存泄漏。

2.2 軟引用

在Java中使用SoftReference幫助我們定義軟引用。其構造方法有兩個:

public SoftReference(T referent);public SoftReference(T referent, ReferenceQueue<? super T> q);

兩個構造方法相似,第二個比第一個多了一個引用隊列,在構造方法中的第一個參數就是我們的實際被指向的對象,這里用新建一個SoftReference來替代我們上面強引用的等號。 下面是構造軟引用的例子:

 softZhangsan = new SoftReference(new User("zhangsan", 24));

2.2.1軟引用有什么用?

如果某個對象他只被軟引用所指向,那么他將會在內存要溢出的時候被回收,也就是當我們要出現OOM的時候,如果回收了一波內存還不夠,這才拋出OOM,弱引用回收的時候如果設置了引用隊列,那么這個軟引用還會進一次引用隊列,但是引用所指向的對象已經被回收。這里要和下面的弱引用區分開來,弱引用是只要有垃圾回收,那么他所指向的對象就會被回收。下面是一個代碼例子:

public static void main(String[] args) { ReferenceQueue<User> referenceQueue = new ReferenceQueue(); SoftReference softReference = new SoftReference(new User("zhangsan",24), referenceQueue); //手動觸發GC System.gc(); Thread.sleep(1000); System.out.println("手動觸發GC:" + softReference.get()); System.out.println("手動觸發的隊列:" + referenceQueue.poll()); //通過堆內存不足觸發GC makeHeapNotEnough(); System.out.println("通過堆內存不足觸發GC:" + softReference.get()); System.out.println("通過堆內存不足觸發GC:" + referenceQueue.poll()); } private static void makeHeapNotEnough() { SoftReference softReference = new SoftReference(new byte[1024*1024*5]); byte[] bytes = new byte[1024*1024*5]; } 輸出: 手動觸發GC:User{name='zhangsan', age=24} 手動觸發的隊列:null 通過堆內存不足觸發GC:null 通過堆內存不足觸發GC:java.lang.ref.SoftReference@4b85612c

通過-Xmx10m設置我們堆內存大小為10,方便構造堆內存不足的情況??梢钥匆娢覀冚敵龅那闆r我們手動調用System.gc并沒有回收我們的軟引用所指向的對象,只有在內存不足的情況下才能觸發。

2.2.2軟應用的應用

在SoftReference的doc中有這么一句話:

Soft references are most often used to implement memory-sensitive caches

也就是說軟引用經常用來實現內存敏感的高速緩存。怎么理解這句話呢?我們知道軟引用他只會在內存不足的時候才觸發,不會像強引用那用容易內存溢出,我們可以用其實現高速緩存,一方面內存不足的時候可以回收,一方面也不會頻繁回收。在高速本地緩存Caffeine中實現了軟引用的緩存,當需要緩存淘汰的時候,如果是只有軟引用指向那么久會被回收。不熟悉Caffeine的同學可以閱讀深入理解Caffeine

2.3 弱引用

弱引用在Java中使用WeakReference來定義一個弱引用,上面我們說過他比軟引用更加弱,只要發生垃圾回收,若這個對象只被弱引用指向,那么就會被回收。這里我們就不多廢話了,直接上例子:

public static void main(String[] args) { WeakReference weakReference = new WeakReference(new User("zhangsan",24)); System.gc(); System.out.println("手動觸發GC:" + weakReference.get()); }輸出結果:手動觸發GC:null

可以看見上面的例子只要垃圾回收一觸發,該對象就被回收了。

2.3.1 弱引用的作用

在WeakReference的注釋中寫到:

Weak references are most often used to implement canonicalizing mappings.

從中可以知道虛引用更多的是用來實現canonicalizing mappings(規范化映射)。在JDK中WeakHashMap很好的體現了這個例子:

public static void main(String[] args) throws Exception { WeakHashMap<User, String> weakHashMap = new WeakHashMap(); //強引用 User zhangsan = new User("zhangsan", 24); weakHashMap.put(zhangsan, "zhangsan"); System.out.println("有強引用的時候:map大小" + weakHashMap.size()); //去掉強引用 zhangsan = null; System.gc(); Thread.sleep(1000); System.out.println("無強引用的時候:map大小"+weakHashMap.size()); }輸出結果為:有強引用的時候:map大小1無強引用的時候:map大小0

可以看出在GC之后我們在map中的鍵值對就被回收了,在weakHashMap中其實只有Key是虛引用做關聯的,然后通過引用隊列再去對我們的map進行回收處理。

2.4 虛引用

虛引用是最弱的引用,在Java中使用PhantomReference進行定義。弱到什么地步呢?也就是你定義了虛引用根本無法通過虛引用獲取到這個對象,更別談影響這個對象的生命周期了。在虛引用中唯一的作用就是用隊列接收對象即將死亡的通知。

 public static void main(String[] args) throws Exception { ReferenceQueue referenceQueue = new ReferenceQueue(); PhantomReference phantomReference = new PhantomReference(new User("zhangsan", 24), referenceQueue); System.out.println("什么也不做,獲取:" + phantomReference.get()); }輸出結果:什么也不做,獲取:null

在PhantomReference的注釋中寫到:

Phantom references are most often used for scheduling pre-mortem cleanup actions in a more flexible way than is possible with the Java finalization mechanism.

虛引用得最多的就是在對象死前所做的清理操作,這是一個比Java的finalization梗靈活的機制。 在DirectByteBuffer中使用Cleaner用來回收對外內存,Cleaner是PhantomReference的子類,當DirectByteBuffer被回收的時候未防止內存泄漏所以通過這種方式進行回收,有點類似于下面的代碼:

public static void main(String[] args) throws Exception { Cleaner.create(new User("zhangsan", 24), () -> {System.out.println("我被回收了,當前線程:{}"+ Thread.currentThread().getName());}); System.gc(); Thread.sleep(1000); }輸出:我被回收了,當前線程:Reference Handler

3 ThreadLocal

ThreadLocal是一個本地線程副本變量工具類,基本在我們的代碼中隨處可見。這里就不過多的介紹他了。

3.1 ThreadLocal和弱引用的那些事

上面說了這么多關于引用的事,這里終于回到了主題了我們的ThreadLocal和弱引用有什么關系呢?

在我們的Thread類中有下面這個變量:

ThreadLocal.ThreadLocalMap threadLocals

ThreadLocalMap本質上也是個Map,其中Key是我們的ThreadLocal這個對象,Value就是我們在ThreadLocal中保存的值。也就是說我們的ThreadLocal保存和取對象都是通過Thread中的ThreadLocalMap來操作的,而key就是本身。在ThreadLocalMap中Entry有如下定義:

 static class Entry extends WeakReference<ThreadLocal<?>> { /** The value associated with this ThreadLocal. */ Object value; Entry(ThreadLocal<?> k, Object v) { super(k); value = v; } }

可以看見Entry是WeakReference的子類,而這個虛引用所關聯的對象正是我們的ThreadLocal這個對象。我們又回到上面的問題:

"threadlocal的key是虛引用,那么在threadlocal.get()的時候,發生GC之后,key是否是null?"
這個問題晃眼一看,虛引用嘛,還有垃圾回收那肯定是為null,這其實是不對的,因為題目說的是在做threadlocal.get()操作,證明其實還是有強引用存在的。所以key并不為null。如果我們的強引用不存在的話,那么Key就會被回收,也就是會出現我們value沒被回收,key被回收,導致value永遠存在,出現內存泄漏。這也是ThreadLocal經常會被很多書籍提醒到需要remove()的原因。

你也許會問看到很多源碼的ThreadLocal并沒有寫remove依然再用得很好呢?那其實是因為很多源碼經常是作為靜態變量存在的生命周期和Class是一樣的,而remove需要再那些方法或者對象里面使用ThreadLocal,因為方法?;蛘邔ο蟮匿N毀從而強引用丟失,導致內存泄漏。

3.2 FastThreadLocal

FastThreadLocal是Netty中提供的高性能本地線程副本變量工具。在Netty的io.netty.util中提供了很多牛逼的工具,后續會一一給大家介紹,這里就先說下FastThreadLocal。

FastThreadLocal有下面幾個特點:

使用數組代替ThreadLocalMap存儲數據,從而獲取更快的性能。(緩存行和一次定位,不會有hash沖突)
由于使用數組,不會出現Key回收,value沒被回收的尷尬局面,所以避免了內存泄漏。

總結

文章開頭的問題,為什么會被問出來,其實是對虛引用和ThreadLocal理解不深導致,很多時候只記著一個如果是虛引用,在垃圾回收時就會被回收,就會導致把這個觀念先入為主,沒有做更多的分析思考。所以大家再分析一個問題的時候還是需要更多的站在不同的場景上做更多的思考。

以上所述是小編給大家介紹的Java引用和Threadlocal的那些事,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對VeVb武林網網站的支持!


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品7m视频| 欧美一级成年大片在线观看| 日韩精品高清视频| 亚洲xxx自由成熟| 日韩大片免费观看视频播放| 亚洲成人激情小说| 国内精品久久久久伊人av| 欧美视频二区36p| 欧美精品久久一区二区| 欧美国产激情18| 欧美一区二粉嫩精品国产一线天| 中文国产亚洲喷潮| 97福利一区二区| 日韩精品视频在线| 亚洲va欧美va国产综合久久| 国产成人福利夜色影视| 在线播放日韩精品| 国产一区二区三区高清在线观看| 久久久天堂国产精品女人| 日韩精品中文字幕有码专区| 欧美一区二区色| 国产精品成人av在线| 成人精品福利视频| www欧美xxxx| 亚洲精品福利免费在线观看| 亚洲国产另类久久精品| 亚洲一区精品电影| 亚洲肉体裸体xxxx137| 欧美最顶级的aⅴ艳星| 岛国av一区二区在线在线观看| 91社影院在线观看| 国产免费久久av| 97激碰免费视频| 国产亚洲欧洲高清| 中文字幕精品一区二区精品| 黑丝美女久久久| 成人有码在线播放| 国产午夜精品久久久| 最近2019免费中文字幕视频三| 国产日韩亚洲欧美| 韩日欧美一区二区| 亚洲欧美在线一区| 国产欧美欧洲在线观看| 亚洲精品美女在线观看| 久久久亚洲影院你懂的| 亚洲国产精品视频在线观看| 国产精品视频一区二区三区四| 在线视频中文亚洲| 精品视频www| 揄拍成人国产精品视频| 国产精品高潮呻吟视频| 裸体女人亚洲精品一区| 亚洲一区二区三区四区视频| 精品国产精品三级精品av网址| 久久久黄色av| 欧美日韩国产中文字幕| 欧美日韩激情视频| 精品久久久一区二区| 一本色道久久88精品综合| 性色av一区二区三区在线观看| 在线观看亚洲区| 国产精品福利观看| 欧美极品xxxx| 亚洲精品久久久久国产| 久久精品2019中文字幕| 超碰91人人草人人干| 日韩欧美精品在线观看| 国产精品91久久久| 亚洲精品美女在线观看播放| 91po在线观看91精品国产性色| 国产精品爱久久久久久久| 国产精品中文久久久久久久| 中文国产成人精品| 在线观看精品国产视频| 欧美黄色片视频| 久久久久www| 日本人成精品视频在线| 高清一区二区三区四区五区| 欧美久久久精品| 日韩在线视频中文字幕| 在线电影欧美日韩一区二区私密| 精品国产31久久久久久| 日韩在线视频免费观看| 91精品啪aⅴ在线观看国产| 色七七影院综合| 在线精品视频视频中文字幕| 亚洲男人天堂2023| 久久97精品久久久久久久不卡| 98精品在线视频| 成人欧美一区二区三区黑人| 精品久久久免费| 中文欧美日本在线资源| 欧美日韩国产综合视频在线观看中文| 亚洲最大福利网| 国产成人激情小视频| 国产suv精品一区二区三区88区| 欧美一区二区三区图| 日韩免费高清在线观看| 欧美二区乱c黑人| 亚洲精品456在线播放狼人| 国产九九精品视频| 欧美国产精品人人做人人爱| 亚洲大胆人体视频| 欧美电影免费观看电视剧大全| 久久久国产精品视频| 亚洲综合日韩中文字幕v在线| 久久艳片www.17c.com| 91av福利视频| 国产不卡av在线免费观看| 欧美老女人在线视频| 国产精品亚洲激情| 亚洲欧美日韩图片| 日韩精品视频在线观看免费| 欧美日韩亚洲视频| 国产精品丝袜久久久久久不卡| 欧洲s码亚洲m码精品一区| 青草青草久热精品视频在线网站| 欧美日韩在线第一页| 亚洲国产婷婷香蕉久久久久久| 一区二区在线视频| 中国日韩欧美久久久久久久久| 国产精品爱啪在线线免费观看| 久久免费视频在线观看| 国产精品久久久久久久7电影| 国产精品日韩欧美大师| 国产成人啪精品视频免费网| 久久久免费观看视频| 国产精品久久久久久久久久三级| 欧美日韩精品中文字幕| 97精品伊人久久久大香线蕉| 久久久精品国产| 国产精品成人观看视频国产奇米| 亚洲免费人成在线视频观看| 国产欧美日韩精品专区| www.亚洲男人天堂| 亚洲欧美日韩精品久久亚洲区| 日韩高清电影免费观看完整| 亚洲国产古装精品网站| 2019av中文字幕| 欧美日韩在线视频首页| 91色在线观看| 亚洲欧美国产日韩天堂区| 亚洲视频免费一区| 欧美激情视频给我| 中文字幕亚洲国产| 久久久久国色av免费观看性色| 欧美高清性猛交| 久久久久久一区二区三区| 亚洲综合日韩在线| 色偷偷av一区二区三区| 九九热在线精品视频| 精品久久久久久久中文字幕| 亚洲性线免费观看视频成熟| 中文字幕日韩欧美精品在线观看| 亚洲精品视频免费在线观看| 亚洲国产精品电影在线观看| 国产欧美婷婷中文| 亚洲成年人影院在线| 亚洲奶大毛多的老太婆| 欧美性在线观看| 中文字幕综合在线| 一区二区中文字幕| 亚洲美女又黄又爽在线观看| 成人精品网站在线观看|