溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中ThreadLocal導致內(nèi)存溢出的原因有哪些

發(fā)布時間:2023-05-09 09:32:08 來源:億速云 閱讀:279 作者:zzz 欄目:編程語言

這篇文章主要講解了“Java中ThreadLocal導致內(nèi)存溢出的原因有哪些”,文中的講解內(nèi)容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Java中ThreadLocal導致內(nèi)存溢出的原因有哪些”吧!

原因分析

ThreadLocal 導致內(nèi)存 OOM 的原因是什么?

ThreadLocal 底層通過 ThreadLocalMap 存儲數(shù)據(jù)

源碼如下:

Java中ThreadLocal導致內(nèi)存溢出的原因有哪些

  • 當我們使用ThreadLocal.set()時,set的value與key(即業(yè)務自己定義的ThreadLocal類)會存儲在ThreadLocalMap的Entry[]數(shù)組里

源碼如下:

Java中ThreadLocal導致內(nèi)存溢出的原因有哪些

  • 其中Entry是實現(xiàn)了一個弱引用WeakReference,Entry的key(即業(yè)務方定義的 ThreadLocal類)會被包裝成一個弱引用當成Entry的key。Java的弱引用的定義是,當JVM執(zhí)行垃圾回收掃描的時候,當發(fā)現(xiàn)只有弱引用的對象時,會立即回收此對象,這是ThreadLocal當初設計的時候防止內(nèi)存溢出的一個手段

源碼如下:

Java中ThreadLocal導致內(nèi)存溢出的原因有哪些

雖然key被包裝成了一個弱引用會被垃圾回收機制給回收,但是value在線程(Thread)不死亡時卻可能存在一條強引用鏈.

由于 value是強引用,只要 Thread不死亡時,例如線程池,這條強引用鏈就會存在,那么value就不會回收,可能造成內(nèi)存溢出

引用關系如下: Thread ==> ThreadLocalMap ==> Entry ==> value

但是這個消除強引用鏈的動作是需要業(yè)務方在get的情況下觸發(fā)的,可能業(yè)務方并不會get、也可能get是key不為空,并不會觸發(fā) expungeStaleEntry 類。所以開發(fā)者要養(yǎng)成良好的習慣,記得用完 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 {
                    // 設置變量值
                    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"}

感謝各位的閱讀,以上就是“Java中ThreadLocal導致內(nèi)存溢出的原因有哪些”的內(nèi)容了,經(jīng)過本文的學習后,相信大家對Java中ThreadLocal導致內(nèi)存溢出的原因有哪些這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI