欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

實(shí)例詳解Java中ThreadLocal內(nèi)存泄露

 更新時(shí)間:2016年08月14日 10:59:16   投稿:daisy  
這一篇文章我們來分析一個(gè)Java中ThreadLocal內(nèi)存泄露的案例。分析問題的過程比結(jié)果更重要,理論結(jié)合實(shí)際才能徹底分析出內(nèi)存泄漏的原因。

案例與分析

問題背景

在 Tomcat 中,下面的代碼都在 webapp 內(nèi),會導(dǎo)致WebappClassLoader泄漏,無法被回收。

public class MyCounter {
    private int count = 0;

    public void increment() {
        count++;
    }

    public int getCount() {
        return count;
    }
}

public class MyThreadLocal extends ThreadLocal<MyCounter> {
}

public class LeakingServlet extends HttpServlet {
    private static MyThreadLocal myThreadLocal = new MyThreadLocal();

    protected void doGet(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException {

        MyCounter counter = myThreadLocal.get();
        if (counter == null) {
            counter = new MyCounter();
            myThreadLocal.set(counter);
        }

        response.getWriter().println(
                "The current thread served this servlet " + counter.getCount()
                        + " times");
        counter.increment();
    }
}

上面的代碼中,只要LeakingServlet被調(diào)用過一次,且執(zhí)行它的線程沒有停止,就會導(dǎo)致WebappClassLoader泄漏。每次你 reload 一下應(yīng)用,就會多一份WebappClassLoader實(shí)例,最后導(dǎo)致 PermGen OutOfMemoryException。

解決問題

現(xiàn)在我們來思考一下:為什么上面的ThreadLocal子類會導(dǎo)致內(nèi)存泄漏?

WebappClassLoader

首先,我們要搞清楚WebappClassLoader是什么鬼?

對于運(yùn)行在 Java EE容器中的 Web 應(yīng)用來說,類加載器的實(shí)現(xiàn)方式與一般的 Java 應(yīng)用有所不同。不同的 Web 容器的實(shí)現(xiàn)方式也會有所不同。以 Apache Tomcat 來說,每個(gè) Web 應(yīng)用都有一個(gè)對應(yīng)的類加載器實(shí)例。該類加載器也使用代理模式,所不同的是它是首先嘗試去加載某個(gè)類,如果找不到再代理給父類加載器。這與一般類加載器的順序是相反的。這是 Java Servlet 規(guī)范中的推薦做法,其目的是使得 Web 應(yīng)用自己的類的優(yōu)先級高于 Web 容器提供的類。這種代理模式的一個(gè)例外是:Java 核心庫的類是不在查找范圍之內(nèi)的。這也是為了保證 Java 核心庫的類型安全。

也就是說WebappClassLoader是 Tomcat 加載 webapp 的自定義類加載器,每個(gè) webapp 的類加載器都是不一樣的,這是為了隔離不同應(yīng)用加載的類。

那么WebappClassLoader的特性跟內(nèi)存泄漏有什么關(guān)系呢?目前還看不出來,但是它的一個(gè)很重要的特點(diǎn)值得我們注意:每個(gè) webapp 都會自己的WebappClassLoader,這跟 Java 核心的類加載器不一樣。

我們知道:導(dǎo)致WebappClassLoader泄漏必然是因?yàn)樗粍e的對象強(qiáng)引用了,那么我們可以嘗試畫出它們的引用關(guān)系圖。等等!類加載器的作用到底是啥?為什么會被強(qiáng)引用?

類的生命周期與類加載器

要解決上面的問題,我們得去研究一下類的生命周期和類加載器的關(guān)系。

跟我們這個(gè)案例相關(guān)的主要是類的卸載:

在類使用完之后,如果滿足下面的情況,類就會被卸載:

      1、該類所有的實(shí)例都已經(jīng)被回收,也就是 Java 堆中不存在該類的任何實(shí)例。

      2、加載該類的ClassLoader已經(jīng)被回收。

      3、該類對應(yīng)的java.lang.Class對象沒有任何地方被引用,沒有在任何地方通過反射訪問該類的方法。

如果以上三個(gè)條件全部滿足,JVM 就會在方法區(qū)垃圾回收的時(shí)候?qū)︻愡M(jìn)行卸載,類的卸載過程其實(shí)就是在方法區(qū)中清空類信息,Java 類的整個(gè)生命周期就結(jié)束了。

由Java虛擬機(jī)自帶的類加載器所加載的類,在虛擬機(jī)的生命周期中,始終不會被卸載。Java虛擬機(jī)自帶的類加載器包括根類加載器、擴(kuò)展類加載器和系統(tǒng)類加載器。Java虛擬機(jī)本身會始終引用這些類加載器,而這些類加載器則會始終引用它們所加載的類的Class對象,因此這些Class對象始終是可觸及的。

由用戶自定義的類加載器加載的類是可以被卸載的。

注意上面這句話,WebappClassLoader如果泄漏了,意味著它加載的類都無法被卸載,這就解釋了為什么上面的代碼會導(dǎo)致 PermGen OutOfMemoryException。

關(guān)鍵點(diǎn)看下面這幅圖

 

我們可以發(fā)現(xiàn):類加載器對象跟它加載的 Class 對象是雙向關(guān)聯(lián)的。這意味著,Class 對象可能就是強(qiáng)引用WebappClassLoader,導(dǎo)致它泄漏的元兇。

引用關(guān)系圖

理解類加載器與類的生命周期的關(guān)系之后,我們可以開始畫引用關(guān)系圖了。(圖中的LeakingServlet.classmyThreadLocal引用畫的不嚴(yán)謹(jǐn),主要是想表達(dá)myThreadLocal是類變量的意思)

下面,我們根據(jù)上面的圖來分析WebappClassLoader泄漏的原因。

      1、LeakingServlet持有staticMyThreadLocal,導(dǎo)致myThreadLocal的生命周期跟LeakingServlet類的生命周期一樣長。意味著myThreadLocal不會被回收,弱引用形同虛設(shè),所以當(dāng)前線程無法通過ThreadLocalMap的防護(hù)措施清除counter的強(qiáng)引用。

      2、強(qiáng)引用鏈:thread -> threadLocalMap -> counter -> MyCounter.class -> WebappClassLocader,導(dǎo)致WebappClassLoader泄漏。

總結(jié)

內(nèi)存泄漏是很難發(fā)現(xiàn)的問題,往往由于多方面原因造成。ThreadLocal由于它與線程綁定的生命周期成為了內(nèi)存泄漏的常客,稍有不慎就釀成大禍。本文只是對一個(gè)特定案例的分析,若能以此舉一反三,那便是極好的。希望本文對大家能有所幫助。

相關(guān)文章

  • Java程序執(zhí)行過程及內(nèi)存機(jī)制詳解

    Java程序執(zhí)行過程及內(nèi)存機(jī)制詳解

    本講將介紹Java代碼是如何一步步運(yùn)行起來的,還會介紹Java程序所占用的內(nèi)存是被如何管理的:堆、棧和方法區(qū)都各自負(fù)責(zé)存儲哪些內(nèi)容,感興趣的朋友跟隨小編一起看看吧
    2020-12-12
  • Java時(shí)間轉(zhuǎn)換成unix時(shí)間戳的方法

    Java時(shí)間轉(zhuǎn)換成unix時(shí)間戳的方法

    這篇文章主要為大家詳細(xì)介紹了Java時(shí)間轉(zhuǎn)換成unix時(shí)間戳的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • Spring Boot 捕捉全局異常 統(tǒng)一返回值的問題

    Spring Boot 捕捉全局異常 統(tǒng)一返回值的問題

    這篇文章主要介紹了Spring Boot 捕捉全局異常 統(tǒng)一返回值,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-06-06
  • MybatisPlus?LambdaQueryWrapper使用int默認(rèn)值的坑及解決

    MybatisPlus?LambdaQueryWrapper使用int默認(rèn)值的坑及解決

    這篇文章主要介紹了MybatisPlus?LambdaQueryWrapper使用int默認(rèn)值的坑及解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
    2022-01-01
  • java socket 詳細(xì)介紹

    java socket 詳細(xì)介紹

    本篇文章小編為大家介紹,java socket 詳細(xì)介紹。需要的朋友參考下
    2013-04-04
  • 利用java操作Excel文件的方法

    利用java操作Excel文件的方法

    以下是對利用java操作Excel文件的方法進(jìn)行了詳細(xì)的介紹,需要的朋友可以過來參考下
    2013-09-09
  • IDEA+Maven創(chuàng)建Spring項(xiàng)目的實(shí)現(xiàn)步驟

    IDEA+Maven創(chuàng)建Spring項(xiàng)目的實(shí)現(xiàn)步驟

    這篇文章主要介紹了IDEA+Maven創(chuàng)建Spring項(xiàng)目的實(shí)現(xiàn)步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • Spring Boot如何讀取自定義外部屬性詳解

    Spring Boot如何讀取自定義外部屬性詳解

    這篇文章主要給大家介紹了關(guān)于Spring Boot如何讀取自定義外部屬性的相關(guān)資料,這個(gè)功能實(shí)現(xiàn)介紹的很詳細(xì),需要的朋友可以參考下
    2021-05-05
  • Spring常用注解及http數(shù)據(jù)轉(zhuǎn)換教程

    Spring常用注解及http數(shù)據(jù)轉(zhuǎn)換教程

    這篇文章主要為大家介紹了Spring常用注解及http數(shù)據(jù)轉(zhuǎn)換原理以及接收復(fù)雜嵌套對象參數(shù)與Http數(shù)據(jù)轉(zhuǎn)換的原理,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2022-03-03
  • 關(guān)于Java中Bean的生命周期詳解

    關(guān)于Java中Bean的生命周期詳解

    這篇文章主要介紹了關(guān)于Java中Bean的生命周期詳解,所謂的?命周期指的是?個(gè)對象從誕?到銷毀的整個(gè)?命過程,我們把這個(gè)過程就叫做?個(gè)對象的?命周期,需要的朋友可以參考下
    2023-08-08

最新評論