Java中ThreadLocal?導(dǎo)致內(nèi)存?OOM?的原因分析
原因分析
ThreadLocal 導(dǎo)致內(nèi)存 OOM 的原因是什么?
ThreadLocal 底層通過 ThreadLocalMap 存儲數(shù)據(jù)
源碼如下:
- 當我們使用ThreadLocal.set()時,set的value與key(即業(yè)務(wù)自己定義的ThreadLocal類)會存儲在ThreadLocalMap的Entry[]數(shù)組里
源碼如下:
- 其中Entry是實現(xiàn)了一個弱引用WeakReference,Entry的key(即業(yè)務(wù)方定義的 ThreadLocal類)會被包裝成一個弱引用當成Entry的key。Java的弱引用的定義是,當JVM執(zhí)行垃圾回收掃描的時候,當發(fā)現(xiàn)只有弱引用的對象時,會立即回收此對象,這是ThreadLocal當初設(shè)計的時候防止內(nèi)存溢出的一個手段
源碼如下:
雖然key被包裝成了一個弱引用會被垃圾回收機制給回收,但是value在線程(Thread)不死亡時卻可能存在一條強引用鏈.
由于 value
是強引用,只要 Thread
不死亡時,例如線程池,這條強引用鏈就會存在,那么value
就不會回收,可能造成內(nèi)存溢出
引用關(guān)系如下: Thread ==> ThreadLocalMap ==> Entry ==> value
但是這個消除強引用鏈的動作是需要業(yè)務(wù)方在get的情況下觸發(fā)的,可能業(yè)務(wù)方并不會get、也可能get是key不為空,并不會觸發(fā) expungeStaleEntry 類。所以開發(fā)者要養(yǎng)成良好的習(xí)慣,記得用完 ThreadLocal
時,調(diào)一次ThreadLocal.remove()
方法或者 ThreadLocal.set(null)
正確的使用方式
public class ThreadLocalTest { /** * 未初始化的本地線程變量 */ private static ThreadLocal<User> userThreadLocal = new ThreadLocal<>(); public static void main(String[] args) throws InterruptedException { int threadNum = 10; List<Thread> threadList = Lists.newArrayList(); for (int i = 0; i < threadNum; ++i) { long userId = i; Thread t = new Thread(() -> { try { // 設(shè)置變量值 userThreadLocal.set(new User(userId, "lanxing" + userId, "2x")); // 使用變量 doSomething(); } finally { // 移除變量 userThreadLocal.remove(); //移除ThreadLocal變量 } }, "T" + i); threadList.add(t); t.start(); } for (int i = 0; i < threadNum; ++i) { threadList.get(i).join(); } } private static void doSomething() { log.info("Use ThreadLocal variable :{}", JSON.toJSONString(userThreadLocal.get())); } } @Data @NoArgsConstructor @AllArgsConstructor class User { private Long id; private String name; private String level; }
打印結(jié)果:
14:30:26.790 [T2] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":2,"level":"2x","name":"lanxing2"}
14:30:26.789 [T5] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":5,"level":"2x","name":"lanxing5"}
14:30:26.792 [T0] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":0,"level":"2x","name":"lanxing0"}
14:30:26.792 [T4] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":4,"level":"2x","name":"lanxing4"}
14:30:26.792 [T8] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":8,"level":"2x","name":"lanxing8"}
14:30:26.791 [T1] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":1,"level":"2x","name":"lanxing1"}
14:30:26.792 [T7] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":7,"level":"2x","name":"lanxing7"}
14:30:26.792 [T6] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":6,"level":"2x","name":"lanxing6"}
14:30:26.791 [T9] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":9,"level":"2x","name":"lanxing9"}
14:30:26.790 [T3] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":3,"level":"2x","name":"lanxing3"}
到此這篇關(guān)于Java中ThreadLocal 導(dǎo)致內(nèi)存 OOM 的原因分析的文章就介紹到這了,更多相關(guān)Java 內(nèi)存OOM 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java中的ThreadPoolExecutor線程池原理細節(jié)解析
這篇文章主要介紹了Java中的ThreadPoolExecutor線程池原理細節(jié)解析,ThreadPoolExecutor是一個線程池,最多可使用7個參數(shù)來控制線程池的生成,使用線程池可以避免創(chuàng)建和銷毀線程的資源損耗,提高響應(yīng)速度,并且可以管理線程池中線程的數(shù)量和狀態(tài)等等,需要的朋友可以參考下2023-12-12SpringBoot Redis緩存數(shù)據(jù)實現(xiàn)解析
這篇文章主要介紹了SpringBoot Redis緩存數(shù)據(jù)實現(xiàn)解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-01-01Springboot中的@ComponentScan注解使用解析
這篇文章主要介紹了Springboot中的@ComponentScan注解使用解析,@ComponentScan用于類或接口上主要是指定掃描路徑,spring會把指定路徑下帶有指定注解的類注冊到IOC容器中,需要的朋友可以參考下2024-01-01java搭建一個Socket服務(wù)器響應(yīng)多用戶訪問
本篇文章主要介紹了java搭建一個Socket服務(wù)器響應(yīng)多用戶訪問,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-02-02java發(fā)送http get請求的兩種方法(總結(jié))
下面小編就為大家?guī)硪黄猨ava發(fā)送http get請求的兩種方法(總結(jié))。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-05-05Java數(shù)據(jù)結(jié)構(gòu)之并查集的實現(xiàn)
并查集是一種用來管理元素分組情況的數(shù)據(jù)結(jié)構(gòu)。并查集可以高效地進行如下操作。本文將通過Java實現(xiàn)并查集,感興趣的小伙伴可以了解一下2022-01-01