精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

ThreadLocal發生內存泄漏的原因

汽車玩家 ? 來源:Java識堂 ? 作者:Java識堂 ? 2020-05-05 16:23 ? 次閱讀

前言

ThreadLocal 的作用是提供線程內的局部變量,這種變量在線程的生命周期內起作用,減少同一個線程內多個函數或者組件之間一些公共變量的傳遞的復雜度。但是如果濫用 ThreadLocal,就可能會導致內存泄漏。下面,我們將圍繞三個方面來分析 ThreadLocal 內存泄漏的問題

ThreadLocal 實現原理

ThreadLocal為什么會內存泄漏

ThreadLocal 最佳實踐

ThreadLocal 實現原理

ThreadLocal的實現是這樣的:每個Thread 維護一個 ThreadLocalMap 映射表,這個映射表的 key 是 ThreadLocal 實例本身,value 是真正需要存儲的 Object。

也就是說 ThreadLocal 本身并不存儲值,它只是作為一個 key 來讓線程從 ThreadLocalMap 獲取 value。值得注意的是圖中的虛線,表示 ThreadLocalMap 是使用 ThreadLocal 的弱引用作為 Key 的,弱引用的對象在 GC 時會被回收。

ThreadLocal為什么會內存泄漏

ThreadLocalMap使用ThreadLocal的弱引用作為key,如果一個ThreadLocal沒有外部強引用來引用它,那么系統 GC 的時候,這個ThreadLocal勢必會被回收,這樣一來,ThreadLocalMap中就會出現key為null的Entry,就沒有辦法訪問這些key為null的Entry的value,如果當前線程再遲遲不結束的話,這些key為null的Entry的value就會一直存在一條強引用鏈:Thread Ref -> Thread -> ThreaLocalMap -> Entry -> value永遠無法回收,造成內存泄漏。

其實,ThreadLocalMap的設計中已經考慮到這種情況,也加上了一些防護措施:在ThreadLocal的get(),set(),remove()的時候都會清除線程ThreadLocalMap里所有key為null的value。

但是這些被動的預防措施并不能保證不會內存泄漏:

使用static的ThreadLocal,延長了ThreadLocal的生命周期,可能導致的內存泄漏(參考ThreadLocal 內存泄露的實例分析)。

分配使用了ThreadLocal又不再調用get(),set(),remove()方法,那么就會導致內存泄漏。

為什么使用弱引用

從表面上看內存泄漏的根源在于使用了弱引用。網上的文章大多著重分析ThreadLocal使用了弱引用會導致內存泄漏,但是另一個問題也同樣值得思考:為什么使用弱引用而不是強引用?

我們先來看看官方文檔的說法:

To help deal with very large and long-lived usages, the hash table entries use WeakReferences for keys.為了應對非常大和長時間的用途,哈希表使用弱引用的 key。

下面我們分兩種情況討論:

key 使用強引用:引用的ThreadLocal的對象被回收了,但是ThreadLocalMap還持有ThreadLocal的強引用,如果沒有手動刪除,ThreadLocal不會被回收,導致Entry內存泄漏。

key 使用弱引用:引用的ThreadLocal的對象被回收了,由于ThreadLocalMap持有ThreadLocal的弱引用,即使沒有手動刪除,ThreadLocal也會被回收。value在下一次ThreadLocalMap調用set,get,remove的時候會被清除。

比較兩種情況,我們可以發現:由于ThreadLocalMap的生命周期跟Thread一樣長,如果都沒有手動刪除對應key,都會導致內存泄漏,但是使用弱引用可以多一層保障:弱引用ThreadLocal不會內存泄漏,對應的value在下一次ThreadLocalMap調用set,get,remove的時候會被清除。

因此,ThreadLocal內存泄漏的根源是:由于ThreadLocalMap的生命周期跟Thread一樣長,如果沒有手動刪除對應key就會導致內存泄漏,而不是因為弱引用。

ThreadLocal 最佳實踐

綜合上面的分析,我們可以理解ThreadLocal內存泄漏的前因后果,那么怎么避免內存泄漏呢?

每次使用完ThreadLocal,都調用它的remove()方法,清除數據。

在使用線程池的情況下,沒有及時清理ThreadLocal,不僅是內存泄漏的問題,更嚴重的是可能導致業務邏輯出現問題。所以,使用ThreadLocal就跟加鎖完要解鎖一樣,用完就清理。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 內存
    +關注

    關注

    8

    文章

    3002

    瀏覽量

    73883
  • 變量
    +關注

    關注

    0

    文章

    613

    瀏覽量

    28329
收藏 人收藏

    評論

    相關推薦

    MOS管泄漏電流的類型和產生原因

    MOS管(金屬氧化物半導體場效應晶體管)的泄漏電流是指在MOS管關斷狀態下,從源極或漏極到襯底之間仍然存在的微弱電流。這些泄漏電流可能對電路的性能和穩定性產生不利影響,因此需要深入了解其類型和產生原因。
    的頭像 發表于 10-10 15:11 ?1105次閱讀

    堆棧和內存的基本知識

    本文主要聊聊關于堆棧的內容。包括堆棧和內存的基本知識。常見和堆棧相關的 bug,如棧溢出,內存泄漏,堆內存分配失敗等。后面介紹軟件中堆棧統計的重要性,以及如何使用工具工具軟件中堆棧使用
    的頭像 發表于 08-29 14:10 ?383次閱讀
    堆棧和<b class='flag-5'>內存</b>的基本知識

    如何檢測內存泄漏

    檢測內存泄漏是軟件開發過程中一項至關重要的任務,它有助于識別和解決那些導致程序占用過多內存資源,從而影響程序性能甚至導致程序崩潰的問題。以下將詳細闡述幾種常見的內存
    的頭像 發表于 07-30 11:50 ?1570次閱讀

    NONOS 1.5.3/1.5.4 SSL內存泄漏原因?

    我已經通過隨附的代碼驗證了當發生 SSL 握手錯誤時,會生成內存泄漏 此外,espconn_reconnect_callback不稱為信令ESPCONN_HANDSHAKE - TCP SSL 握手
    發表于 07-18 07:24

    使用system_show_malloc()檢查內存泄漏遇到異常怎么解決?

    我想使用system_show_malloc()檢查內存泄漏,但是當我調用該函數時,我得到了致命的異常: 致命異常 28 (LoadProhibitedCause): epc1
    發表于 07-10 06:32

    高壓電容柜合閘時發生爆炸的原因

    高壓電容柜合閘時發生爆炸是一種嚴重的安全事故,常常給人們的生命財產帶來巨大的損失。了解爆炸的原因,對于預防此類事故的發生至關重要。
    的頭像 發表于 05-28 14:17 ?2w次閱讀
    高壓電容柜合閘時<b class='flag-5'>發生</b>爆炸的<b class='flag-5'>原因</b>

    C語言內存泄漏問題原理

    內存泄漏問題只有在使用堆內存的時候才會出現,棧內存不存在內存泄漏問題,因為棧
    發表于 03-19 11:38 ?482次閱讀
    C語言<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>問題原理

    【鴻蒙】webview內存泄漏問題的分析報告

    內存會隨著使用而不斷增大,最終導致瀏覽器 APP 因內存泄漏而崩潰。 3 問題原因 3.1 正常機制 在任意版本上使用瀏覽器 APP,可以長時間正常瀏覽網頁。 3.2 異常機制 在
    的頭像 發表于 03-02 15:12 ?2079次閱讀

    CYBLE-416045-02在微控制器上制作應用程序,過段時間后突然停止工作的原因?

    我正在使用 CYBLE-416045-02 在微控制器上制作應用程序。 但是,它在 10 分鐘或更長時間后突然停止工作。 我懷疑運行應用程序時是否會發生內存泄漏。 因此,我想檢查是否發生
    發表于 01-22 06:35

    內存溢出與內存泄漏:定義、區別與解決方案

    與區別 1. 定義: 內存溢出(Memory Overflow)指的是程序在申請內存時,無法獲得足夠的內存空間,導致程序拋出異常或崩潰。當程序需要的內存超過了當前可用的
    的頭像 發表于 12-19 14:10 ?2474次閱讀

    嵌入式軟件內存與指針相關問題

    的慢性病,不會立即展現,但不知道未來的哪一天,所有的設備,會在相近的時間點爆發問題。 內存泄漏還會導致系統意外的重啟,重啟的原因可能千奇百怪。因此,檢測和解決內存
    的頭像 發表于 12-07 16:07 ?414次閱讀

    內存泄漏有什么影響

    用malloc申請的內存,沒有釋放,為什么程序也不會出錯?
    的頭像 發表于 12-07 09:42 ?593次閱讀
    <b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>有什么影響

    oom異常的原因和解決方法

    一、OOM異常的原因 OOM異常的出現通常是由于以下幾個原因造成的: 1.1 內存泄漏 內存泄漏
    的頭像 發表于 12-05 13:45 ?6376次閱讀

    Java oom異常的原因分析

    據,而棧內存用于存儲方法調用和局部變量。 當程序需要使用更多內存時,會向操作系統請求更多的內存空間。如果操作系統無法分配足夠的內存空間,就會導致OOM異常的
    的頭像 發表于 12-05 13:43 ?748次閱讀

    jvm內存溢出該如何定位解決

    在Java應用程序中,JVM(Java虛擬機)內存溢出是指Java應用程序試圖分配的內存超過了JVM所允許的最大內存大小,導致程序無法正常執行。內存溢出通常是由以下幾個
    的頭像 發表于 12-05 11:05 ?1292次閱讀