JVM Finalizer線程踩坑記錄

一、finalize方法是對(duì)象回收前的唯一自我救贖機(jī)會(huì)

JVM進(jìn)行GC時(shí),首先使用可達(dá)性分析算法,找出不在GC Roots引用鏈上的對(duì)象,這時(shí)進(jìn)行一次標(biāo)記(標(biāo)記出需要回收的對(duì)象)并篩選(對(duì)需要回收對(duì)象進(jìn)行篩選),篩選條件就是是否有必要執(zhí)行finalize方法。當(dāng)對(duì)象沒(méi)有覆蓋或已執(zhí)行過(guò)finalize方法,則沒(méi)有必要執(zhí)行;否則,將對(duì)象放到由JVM創(chuàng)建的Finalizer線程維護(hù)的F-Queue(java.lang.ref.Finalizer.ReferenceQueue)隊(duì)列中,F(xiàn)inalizer線程會(huì)遍歷執(zhí)行隊(duì)列中對(duì)象的finalize方法,只有當(dāng)F-Queue中對(duì)象finalize執(zhí)行完成后,并且下次GC時(shí)可達(dá)性分析不再GC Roots的引用鏈上,則這些對(duì)象占用的內(nèi)存才能被真正回收。重寫finalize方法可以方便我們?nèi)ブ匦陆?duì)象的引用關(guān)系,避免被回收。

二、多線程環(huán)境重寫對(duì)象的finalize方法

由于Finalizer線程優(yōu)先級(jí)相較于普通線程優(yōu)先級(jí)要低,而根據(jù)Java的搶占式線程調(diào)度策略,優(yōu)先級(jí)越低的線程,分配CPU的機(jī)會(huì)越少,因此當(dāng)多線程創(chuàng)建重寫finalize方法的對(duì)象時(shí),F(xiàn)inalizer可能無(wú)法及時(shí)執(zhí)行finalize方法,F(xiàn)inalizer線程處理對(duì)象的速度小于創(chuàng)建對(duì)象的速度時(shí),會(huì)造成F-Queue越來(lái)越大,JVM內(nèi)存無(wú)法及時(shí)釋放,造成頻繁的Young GC,然后是Full GC,乃至最終的OutOfMemoryError。

三、代理池項(xiàng)目Finalizer線程踩坑記錄

我的個(gè)人爬蟲(chóng)代理池項(xiàng)目中使用多線程+Socket進(jìn)行代理的有效性檢測(cè),代碼如下:

protected static boolean proxyAvailable(Proxy proxy) {
        Socket socket = null;
        if (proxy != null) {
            try {
                if (ProxyUtil.isBasedHttp(proxy)) {
                    socket = new Socket();
                } else {
                    socket = (SSLSocket) ((SSLSocketFactory)SSLSocketFactory.getDefault()).createSocket();
                }
                socket.connect(new InetSocketAddress(proxy.getHost(), proxy.getPort()), 3000);
                return true;
            } catch (IOException e) {
                // do nothing.
            } finally {
                try {
                    socket.close();
                } catch (IOException e) {
                }
            }
        }
        return false;
    }

代理池跑了一段時(shí)間,發(fā)現(xiàn)可用代理越來(lái)越少,看了下GC情況,發(fā)現(xiàn)JVM進(jìn)行了上千次的Full GC,而且堆內(nèi)存基本上占滿了,于是就導(dǎo)出了Javacore和dump分析,在dump里發(fā)現(xiàn)Finalizer線程持有的java.lang.ref.Finalizer.ReferenceQueue里全是java.net.SocksSocketImpl的對(duì)象,所以就把目光投在了上面這一段代碼,跟蹤Socket的源代碼,發(fā)現(xiàn)在創(chuàng)建Socket實(shí)例的時(shí)候,會(huì)調(diào)用這個(gè)方法

    /**
     * Sets impl to the system-default type of SocketImpl.
     * @since 1.4
     */
    void setImpl() {
        if (factory != null) {
            impl = factory.createSocketImpl();
            checkOldImpl();
        } else {
            // No need to do a checkOldImpl() here, we know it's an up to date
            // SocketImpl!
            impl = new SocksSocketImpl();
        }
        if (impl != null)
            impl.setSocket(this);
    }

這里創(chuàng)建了SocksSocketImpl對(duì)象,是系統(tǒng)默認(rèn)的SocketImpl實(shí)現(xiàn)類,而SocksSocketImpl的父類java.net.PlainSocketImpl.PlainSocketImpl的父類java.net.AbstractPlainSocketImpl重寫了finalize方法,在方法里調(diào)用close方法:

    /**
     * Cleans up if the user forgets to close it.
     */
    protected void finalize() throws IOException {
        close();
    }

所以,到這里,問(wèn)題就可以定位了,多線程環(huán)境下,代理檢測(cè)代碼執(zhí)行完成后,Socket對(duì)象被回收,但是,因?yàn)镴VM在回收對(duì)象之前,需要對(duì)象的父類的終止邏輯也要被執(zhí)行,因此,在回收SocksSocketImpl對(duì)象時(shí)需要先執(zhí)行AbstractPlainSocketImpl的finalize方法,我們上面也說(shuō)了,F(xiàn)inalizer線程執(zhí)行優(yōu)先級(jí)低于普通線程,而代理池工程有140個(gè)有效性檢測(cè)線程,對(duì)象銷毀速度趕不上對(duì)象的創(chuàng)建速度,因此,F(xiàn)-Queue越來(lái)越大,JVM瘋狂GC,系統(tǒng)越來(lái)越不可用。

四、代理池優(yōu)化方案

待定,后續(xù)補(bǔ)充

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 原文閱讀 前言 這段時(shí)間懈怠了,罪過(guò)! 最近看到有同事也開(kāi)始用上了微信公眾號(hào)寫博客了,挺好的~給他們點(diǎn)贊,這博客我...
    碼農(nóng)戲碼閱讀 6,144評(píng)論 2 31
  • 1.什么是垃圾回收? 垃圾回收(Garbage Collection)是Java虛擬機(jī)(JVM)垃圾回收器提供...
    簡(jiǎn)欲明心閱讀 90,349評(píng)論 17 311
  • 從三月份找實(shí)習(xí)到現(xiàn)在,面了一些公司,掛了不少,但最終還是拿到小米、百度、阿里、京東、新浪、CVTE、樂(lè)視家的研發(fā)崗...
    時(shí)芥藍(lán)閱讀 42,787評(píng)論 11 349
  • 轉(zhuǎn)自:https://yq.aliyun.com/articles/2947?spm=0.0.0.0.At14xp...
    YDDMAX_Y閱讀 593評(píng)論 0 0
  • 約伯是一個(gè)靠撿垃圾為生的窮老頭,晚上蓋著從垃圾堆撿來(lái)的一床破被子睡覺(jué),白天就在垃圾堆附近找飯吃。每次撿到發(fā)霉的剩面...
    常非常K閱讀 697評(píng)論 4 3

友情鏈接更多精彩內(nèi)容