前言
在生產(chǎn)環(huán)境中甲捏,你是否遇到過(guò)這樣的問(wèn)題磕蛇?
1)系統(tǒng)運(yùn)行一段時(shí)間后old區(qū)空間正逐漸減少命锄?
2)遇到秒殺促銷(xiāo)活動(dòng)等場(chǎng)景old區(qū)存儲(chǔ)突然暴增钾唬,導(dǎo)致JVM Full GC万哪。fgc time持續(xù)過(guò)長(zhǎng)導(dǎo)致cpu 100%?
3)jvm eden區(qū)分配的內(nèi)存實(shí)際并沒(méi)有按照配置的指定抡秆?survivor區(qū)對(duì)象晉升到old區(qū)并沒(méi)有達(dá)到默認(rèn)的15次奕巍?
4)如何避免Full GC,JVM產(chǎn)生FGC后如何解決儒士?
……
接下來(lái)的內(nèi)容的止,我將從實(shí)際場(chǎng)景出發(fā),分別使用三種垃圾收集器:ParallelGC,Concurrent Mark Sweep(cms),G1着撩,進(jìn)行參數(shù)調(diào)優(yōu)诅福,解決我們?cè)谏a(chǎn)中jvm遇到的各種問(wèn)題。jvm調(diào)優(yōu)會(huì)作為專(zhuān)題持續(xù)更新拖叙。
以下GC調(diào)優(yōu)默認(rèn)使用jdk1.8
JVM性能調(diào)優(yōu)實(shí)戰(zhàn)——Concurrent Mark Sweep
jdk1.8默認(rèn)使用ParallelGC氓润。新生代采用的是Parallel Scavenge,老年代Parallel Old薯鳍。
并發(fā)垃圾收集器調(diào)優(yōu)的內(nèi)容一般為:
1)關(guān)閉jvm自動(dòng)分配策略咖气。
2)survivior空間調(diào)優(yōu)。
通過(guò)這兩點(diǎn)調(diào)整,使創(chuàng)建的對(duì)象按照設(shè)定的閾值執(zhí)行崩溪。
-
關(guān)閉JVM自動(dòng)分配策略
先來(lái)看默認(rèn)情況下jvm內(nèi)存空間的分配情況:
-server -Xms4G -Xmx4G -Xss512K -XX:+PrintGCDetails -XX:+PrintGCDateStamps -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/deploy_service_heap.log -verbose:gc -Xloggc:/deploy_service_gc.log
我們使用jdk1.8自帶的jvisualvm工具監(jiān)控查看運(yùn)行情況浅役,如圖:
tips:對(duì)于并發(fā)類(lèi)型的GC來(lái)說(shuō),jvm默認(rèn)開(kāi)啟了內(nèi)存自動(dòng)適配策略參數(shù)是UseAdaptiveSizePolicy悯舟。使用-XX:-UseAdaptiveSizePolicy來(lái)關(guān)閉jvm的自動(dòng)適配策略担租。
-
survivior空間調(diào)優(yōu)
通常的網(wǎng)文會(huì)說(shuō)-XX:SurvivorRatio這個(gè)參數(shù)默認(rèn)為8,意思是eden:s1:s2 = 8:1:1抵怎。這個(gè)比值到底要不要改奋救,修改后效果就一定好嗎?答案是否定的反惕。舉例說(shuō)明:如果改成6(6:2:2)或者4(4:3:3)增加了survivor空間的同時(shí)縮減了eden區(qū)空間尝艘,這會(huì)導(dǎo)致eden區(qū)由于內(nèi)存分配不夠ygc的頻次增加,每ygc一次姿染,存活的對(duì)象就會(huì)進(jìn)入survivor,但實(shí)際情況是survivor中的對(duì)象會(huì)很少背亥,作者的應(yīng)用程序qps200~500之間,持續(xù)時(shí)間為24小時(shí)悬赏,但survivor中的對(duì)象總量最大不超過(guò)70M狡汉,所以擴(kuò)大survivor區(qū)空間是在浪費(fèi)資源。
survivor區(qū)真正要調(diào)整的是TargetSurvivorRatio,MaxTenuringThreshold這兩個(gè)參數(shù)闽颇。
TargetSurvivorRatio:表示目標(biāo)survivor區(qū)存儲(chǔ)率超過(guò)指定百分比時(shí)盾戴,會(huì)重新計(jì)算一次TenuringThreshold值。在生產(chǎn)中如果你遇到這種情況兵多,說(shuō)明需要調(diào)整TargetSurvivorRatio了尖啡。如圖:
圖中survivor區(qū)在每次ygc后都會(huì)重新計(jì)算一次threshold的值(默認(rèn)最大為15),說(shuō)明在survivor區(qū)中的對(duì)象并不都是經(jīng)過(guò)15次以上的ygc才進(jìn)入老年代剩膘。
tips:在并發(fā)類(lèi)型的GC中衅斩,-XX:-UseAdaptiveSizePolicy保證新生代內(nèi)存分配按照你指定的參數(shù)執(zhí)行。-XX:TargetSurvivorRatio=80 -XX:MaxTenuringThreshold=15 讓目標(biāo)survivor使用空間達(dá)到指定的百分比才會(huì)重新計(jì)算threshold值怠褐。如此一來(lái)畏梆,survivor中大部分的對(duì)象會(huì)活到15歲以上才進(jìn)入老年代。優(yōu)點(diǎn)是:阻止新對(duì)象快速進(jìn)入old區(qū)將其填滿(mǎn)奈懒,導(dǎo)致old區(qū)空間使用率100%而觸發(fā)full gc奠涌。
通過(guò)上述調(diào)整后,我們來(lái)看一下jvm運(yùn)行的效果:
我們看到eden區(qū)實(shí)際內(nèi)存分配為1.6G筐赔,survivor區(qū)為204.5M铣猩,并且survivor區(qū)內(nèi)的對(duì)象全部都在第16次ygc后進(jìn)入了old區(qū),這才是我們想要的效果茴丰!
另外補(bǔ)充說(shuō)明一下达皿,ParallelGC優(yōu)調(diào)當(dāng)然遠(yuǎn)不止這些天吓,比如還有一些協(xié)助參數(shù):-XX:+PrintTenuringDistribution 這個(gè)在jvm啟動(dòng)后每次進(jìn)行內(nèi)存分配可以打印出詳細(xì)的內(nèi)存分配情況。
以上是作者通過(guò)實(shí)際生產(chǎn)監(jiān)控發(fā)現(xiàn)的問(wèn)題峦椰,并整理出一些容易被我們忽略的問(wèn)題點(diǎn)龄寞。歡迎各位讀者評(píng)論區(qū)留言進(jìn)行經(jīng)驗(yàn)分享和探討。