在使用log4j2打日志時(shí),當(dāng)發(fā)生大量異常時(shí),造成大量線程block問(wèn)題的問(wèn)題。
大量線程block原因
發(fā)生異常,打印異常棧時(shí),會(huì)調(diào)用org.apache.logging.log4j.core.impl.ThrowableProxy.toExtendedStackTrace方法。
ThrowableProxy.toExtendedStackTrace內(nèi)部會(huì)進(jìn)行l(wèi)oadClass操作。
并且可以看到ClassLoader的loadClass在加載類時(shí)
1)首先會(huì)持有鎖。2)調(diào)用findLoadedClass看下是否類已經(jīng)被加載過(guò)了
3)如果類沒(méi)被加載過(guò),根據(jù)雙親委派模型去加載類。
可以看到當(dāng)某個(gè)類被加載過(guò)了,調(diào)用findLoadedClass會(huì)直接返回,鎖也會(huì)被很快釋放掉,無(wú)需經(jīng)過(guò)雙親委派等后面的一系列步驟。
但是,在進(jìn)行反射調(diào)用時(shí),JVM會(huì)進(jìn)行優(yōu)化,會(huì)動(dòng)態(tài)生成名為sun.reflect.GeneratedMethodAccessor的類,這個(gè)類無(wú)法通過(guò)ClassLoader.loadClass方法加載(為什么無(wú)法通過(guò)ClassLoader.loadClass加載?因?yàn)镴VM內(nèi)部自定義一個(gè)加載器DelegatingClassLoader來(lái)加載這個(gè)類,這導(dǎo)致應(yīng)用類加載器 Launcher$AppClassLoader找不到它)。
導(dǎo)致每次解析異常棧進(jìn)行類加載時(shí),鎖占有的時(shí)間很長(zhǎng),最終導(dǎo)致阻塞。
關(guān)于JVM對(duì)反射調(diào)用的優(yōu)化
Java中對(duì)反射的優(yōu)化
使用反射調(diào)用某個(gè)類的方法,jvm內(nèi)部有兩種方式
JNI:使用native方法進(jìn)行反射操作。
pure-Java:生成bytecode進(jìn)行反射操作,即生成類sun.reflect.GeneratedMethodAccessor,它是一個(gè)被反射調(diào)用方法的包裝類,代理不同的方法,類后綴序號(hào)會(huì)遞增。這種方式第一次調(diào)用速度較慢,較之第一種會(huì)慢3-4倍,但是多次調(diào)用后速度會(huì)提升20倍
對(duì)于使用JNI的方式,因?yàn)槊看味家{(diào)用native方法再返回,速度會(huì)比較慢。所以,當(dāng)一個(gè)方法被反射調(diào)用的次數(shù)超過(guò)一定次數(shù)(默認(rèn)15次)時(shí),JVM內(nèi)部會(huì)進(jìn)行優(yōu)化,使用第2種方法,來(lái)加快運(yùn)行速度。
JVM有兩個(gè)參數(shù)來(lái)控制這種優(yōu)化
-Dsun.reflect.inflationThreshold=<value>value默認(rèn)為15,即反射調(diào)用某個(gè)方法15次后,會(huì)由JNI的方式變?yōu)閜ure-java的方式
-Dsun.reflect.noInflation=true
默認(rèn)為false。當(dāng)設(shè)置為true時(shí),表示在第一次反射調(diào)用時(shí),就轉(zhuǎn)為pure-java的方式
下面是一個(gè)驗(yàn)證反射優(yōu)化的樣例:
配置如下JVM參數(shù),使得在第一次反射調(diào)用時(shí),就轉(zhuǎn)為pure-java的方式
打斷點(diǎn)跟蹤:
可以看到GeneratedMethodAccessor1的classLoader為DelegatingClassLoader,其parent為AppClassLoader。
如何關(guān)閉JVM對(duì)反射調(diào)用的優(yōu)化?
想關(guān)閉JVM對(duì)反射優(yōu)化怎么辦?
JVM中只提供了兩個(gè)參數(shù),因此,沒(méi)有辦法完全關(guān)閉反射優(yōu)化。
一種能想到的接近于關(guān)閉反射優(yōu)化的方法就是將inflationThreshold設(shè)為的一個(gè)特別大的數(shù)。
inflationThreshold是java中的int型值,可以考慮把其設(shè)置為Integer.MAX_VALUE ((2^31)-1)。
$ java-Dsun.reflect.inflationThreshold=2147483647MyApp
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。
新聞名稱:JVM反調(diào)調(diào)用優(yōu)化,導(dǎo)致發(fā)生大量異常時(shí)log4j2線程阻塞-創(chuàng)新互聯(lián)
文章位置:http://www.rwnh.cn/article36/jogpg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供動(dòng)態(tài)網(wǎng)站、商城網(wǎng)站、網(wǎng)站策劃、網(wǎng)站收錄、網(wǎng)站內(nèi)鏈、網(wǎng)站維護(hù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容