久久久久久久av_日韩在线中文_看一级毛片视频_日本精品二区_成人深夜福利视频_武道仙尊动漫在线观看

Java中ThreadLocal?導致內存?OOM?的原因分析

這篇文章主要介紹了Java中ThreadLocal導致內存OOM的原因分析,文章基于Java的相關內容展開ThreadLocal導致內存OOM的原因分析,需要的小伙v阿布可以參考一下

原因分析

ThreadLocal 導致內存 OOM 的原因是什么?

ThreadLocal 底層通過 ThreadLocalMap 存儲數據

源碼如下: 

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

源碼如下:

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

源碼如下: 

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

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

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

但是這個消除強引用鏈的動作是需要業務方在get的情況下觸發的,可能業務方并不會get、也可能get是key不為空,并不會觸發 expungeStaleEntry 類。所以開發者要養成良好的習慣,記得用完 ThreadLocal 時,調一次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;
}

打印結果:

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 導致內存 OOM 的原因分析的文章就介紹到這了,更多相關Java 內存OOM 內容請搜索html5模板網以前的文章希望大家以后多多支持html5模板網!

【網站聲明】本站部分內容來源于互聯網,旨在幫助大家更快的解決問題,如果有圖片或者內容侵犯了您的權益,請聯系我們刪除處理,感謝您的支持!

相關文檔推薦

主站蜘蛛池模板: 欧美aⅴ| 久久久久久亚洲精品 | 精品国产欧美日韩不卡在线观看 | 欧美福利精品 | 精品久久香蕉国产线看观看亚洲 | 在线成人免费视频 | 岛国av一区二区 | 亚洲欧美综合精品另类天天更新 | 亚洲精品福利视频 | 99re热精品视频 | 国产精品色 | 99色综合| 国产精品我不卡 | 国产综合av| 日韩精品一区二区三区视频播放 | 久久99精品久久久久久狂牛 | 黑人巨大精品欧美一区二区免费 | 欧美精品一区二区三区在线播放 | 国产成人99久久亚洲综合精品 | 日本激情一区二区 | 久久久www成人免费无遮挡大片 | 成人精品一区二区三区中文字幕 | 日韩色图视频 | 美女视频黄的免费 | 操人视频在线观看 | 国产亚洲一区二区三区 | 九九久久精品 | av一区在线 | 欧美一区二区三区 | 色爱区综合 | 国产精品二区三区 | 国产日韩欧美精品一区二区三区 | 亚洲激情专区 | 欧美一级在线 | 亚洲精品18 | 青青草视频网 | 伊人超碰 | 亚洲www | h视频免费看 | 国产免费av在线 | 精品欧美乱码久久久久久 |