這篇文章主要講解了“Java中ThreadLocal導(dǎo)致內(nèi)存溢出的原因有哪些”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Java中ThreadLocal導(dǎo)致內(nèi)存溢出的原因有哪些”吧!
拉孜ssl適用于網(wǎng)站、小程序/APP、API接口等需要進(jìn)行數(shù)據(jù)傳輸應(yīng)用場景,ssl證書未來市場廣闊!成為創(chuàng)新互聯(lián)公司的ssl證書銷售渠道,可以享受市場價(jià)格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:18982081108(備注:SSL證書合作)期待與您的合作!
ThreadLocal 導(dǎo)致內(nèi)存 OOM 的原因是什么?
ThreadLocal 底層通過 ThreadLocalMap 存儲數(shù)據(jù)
源碼如下:
當(dāng)我們使用ThreadLocal.set()時(shí),set的value與key(即業(yè)務(wù)自己定義的ThreadLocal類)會存儲在ThreadLocalMap的Entry[]數(shù)組里
源碼如下:
其中Entry是實(shí)現(xiàn)了一個(gè)弱引用WeakReference,Entry的key(即業(yè)務(wù)方定義的 ThreadLocal類)會被包裝成一個(gè)弱引用當(dāng)成Entry的key。Java的弱引用的定義是,當(dāng)JVM執(zhí)行垃圾回收掃描的時(shí)候,當(dāng)發(fā)現(xiàn)只有弱引用的對象時(shí),會立即回收此對象,這是ThreadLocal當(dāng)初設(shè)計(jì)的時(shí)候防止內(nèi)存溢出的一個(gè)手段
源碼如下:
雖然key被包裝成了一個(gè)弱引用會被垃圾回收機(jī)制給回收,但是value在線程(Thread)不死亡時(shí)卻可能存在一條強(qiáng)引用鏈.
由于 value
是強(qiáng)引用,只要 Thread
不死亡時(shí),例如線程池,這條強(qiáng)引用鏈就會存在,那么value
就不會回收,可能造成內(nèi)存溢出
引用關(guān)系如下:Thread ==> ThreadLocalMap ==> Entry ==> value
但是這個(gè)消除強(qiáng)引用鏈的動作是需要業(yè)務(wù)方在get的情況下觸發(fā)的,可能業(yè)務(wù)方并不會get、也可能get是key不為空,并不會觸發(fā) expungeStaleEntry 類。所以開發(fā)者要養(yǎng)成良好的習(xí)慣,記得用完 ThreadLocal
時(shí),調(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"}
感謝各位的閱讀,以上就是“Java中ThreadLocal導(dǎo)致內(nèi)存溢出的原因有哪些”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Java中ThreadLocal導(dǎo)致內(nèi)存溢出的原因有哪些這一問題有了更深刻的體會,具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識點(diǎn)的文章,歡迎關(guān)注!
網(wǎng)站欄目:Java中ThreadLocal導(dǎo)致內(nèi)存溢出的原因有哪些
當(dāng)前網(wǎng)址:http://aaarwkj.com/article16/ihhddg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機(jī)、手機(jī)網(wǎng)站建設(shè)、云服務(wù)器、品牌網(wǎng)站建設(shè)、域名注冊、微信公眾號
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)