JVM性能調(diào)優(yōu)的評(píng)估指標(biāo)及調(diào)優(yōu)示例

上一篇 <<<FullGC、MinorGC父晶、STW等常見(jiàn)問(wèn)題如何解答
下一篇 >>>Class文件分析一個(gè)類為啥最多支持65535個(gè)接口


整體評(píng)估指標(biāo)說(shuō)明

1.吞吐量: 運(yùn)行用戶代碼占總時(shí)間的比例
總運(yùn)行時(shí)間:用戶線程程序的運(yùn)行時(shí)間(100s)+GC內(nèi)存回收的時(shí)間 (1s)
比如程序運(yùn)行時(shí)間100s/內(nèi)存回收時(shí)間 垃圾回收1s 則吞吐量為100/101=99%
2.GC負(fù)荷:與吞吐量相反曲梗,指應(yīng)用花在GC上的時(shí)間百分比
上例GC負(fù)荷為:1/101=1%
3.暫停時(shí)間:應(yīng)用線程花在GC stop-the-world 的時(shí)間
暫時(shí)時(shí)間越小越好
4.GC頻率:次數(shù)/GC頻率越多赞警,stw暫停時(shí)間越短;GC回收頻率次數(shù)越少虏两、stw暫停時(shí)間越長(zhǎng)
5.反應(yīng)速度:從一個(gè)對(duì)象變成垃圾道這個(gè)對(duì)象被回收的時(shí)間
吞吐量?jī)?yōu)先的收集器:Parallel并行收集器【Jdk8默認(rèn)收集器】
響應(yīng)時(shí)間優(yōu)先的收集器:CMS(老年代)/ParNew(新生代)-注重stw時(shí)間越少
G1/ZGC同時(shí)注重吞吐量和響應(yīng)時(shí)間優(yōu)先

JVM調(diào)優(yōu)方案

優(yōu)化核心思路:
1愧旦、通過(guò)堆內(nèi)存設(shè)置減少老年代垃圾回收的次數(shù)
2、配置垃圾回收器定罢,減少STW的時(shí)間

1.避免用戶線程暫停時(shí)間STW比較短

a.堆內(nèi)存空間一定要充足笤虫,垃圾回收和最大堆內(nèi)存無(wú)關(guān),只和初始內(nèi)存有關(guān)祖凫。
b.項(xiàng)目啟動(dòng)堆內(nèi)存初始值與最大值一定保持一致琼蚯,可減少垃圾回收的次數(shù),提高吞吐量;
c.不建議調(diào)用System.gc()惠况,容易造成STW;
d.不要在堆內(nèi)存中存放大對(duì)象和全局變量遭庶,容易觸發(fā)fullgc
e.合理根據(jù)項(xiàng)目堆內(nèi)存情況,選擇收集器

2.合理設(shè)定堆的初始大小和選擇合理的垃圾收集器

a.起步階段的個(gè)人網(wǎng)站,建議堆內(nèi)存1GB 可以串行SerialGC,建議使用并行Parallel GC
b.有一定訪問(wèn)量的網(wǎng)站或APP,建議堆內(nèi)存2g 建議使用Parallel GC
c.并發(fā)適中的APP或普通數(shù)據(jù)處理,建議堆內(nèi)存4g 老年代CMS/新生代parnew
d.適用于并發(fā)要求較高的APP,建議堆內(nèi)存8G(要16G的可以集群)建議G1收集器 注重低延遲和吞吐量

必填參數(shù)

-Xmx 堆最大可用值 測(cè)試結(jié)果:默認(rèn)4G----物理內(nèi)存的1/4
-Xms 堆初始值 測(cè)試結(jié)果:最大內(nèi)存的1/16-----物理內(nèi)存的1/64
-XX:+HeapDumpOnOutOfMemoryError 內(nèi)存溢出的時(shí)候打印內(nèi)存快照
-XX:HeapDumpPath=hdpserver_oom.hprof 內(nèi)存溢出的時(shí)候內(nèi)存快照保存路徑

tips:
a稠屠、-Xmx和-Xms在內(nèi)存不大的時(shí)候建議相同峦睡,內(nèi)存很大的時(shí)候,建議配成2:1
b权埠、內(nèi)存溢出的快照配置很可能幾個(gè)月不出現(xiàn)榨了,但建議配置上,方便問(wèn)題出現(xiàn)時(shí)的排查
c攘蔽、垃圾回收器在JDK8之前建議用CMS龙屉,JDK8及之后建議用G1

優(yōu)化參數(shù)

-XX:+PrintGC 每次觸發(fā)GC的時(shí)候打印相關(guān)日志
-XX:+PrintGCDetails 更詳細(xì)的GC日志
堆設(shè)置
-Xmn 新生代堆最大可用值 默認(rèn)是堆的1/3满俗,官方推薦配置為整個(gè)堆的3/8
-XX:NewRatio 配置新生代與老年代占比 默認(rèn)1:2 建議值:1:2或1:3 -XX:NewRatio=3
-XX:SurvivorRatio 用來(lái)設(shè)置新生代中eden空間和from/to空間的比例=eden/from=den/to 默認(rèn)8:1
-XX:PermSize 初始化永久內(nèi)存區(qū)域大小转捕,默認(rèn)4M,超出的話可能出現(xiàn)PermGen space錯(cuò)誤漫雷。
-XX:MaxPermSize 設(shè)置永久內(nèi)存區(qū)域最大大小
-XX:NewSize 作用跟-XX:NewRatio相似瓜富,不同的是精確的數(shù)值
-XX:MaxNewSize 設(shè)置最大Java新對(duì)象生產(chǎn)堆內(nèi)存,
NewSize和MaxNewSize最好設(shè)成一致降盹,數(shù)值都是1024的整數(shù)倍并且大于1MB与柑。
-XX:MaxTenuringThreshold 設(shè)置垃圾最大年齡
-XX:GCTimeRatio 設(shè)置垃圾回收時(shí)間占程序運(yùn)行時(shí)間的百分比,垃圾回收時(shí)間占程序運(yùn)行時(shí)間百分比的公式為1/(1+n) 蓄坏,如果n=19表示java可以用5%的時(shí)間來(lái)做垃圾回收价捧,1/(1+19)=1/20=5%。
-Xss 我們線程椢写粒空間大小 –Xss1m结蟋,jdk5.0前是256k---配置不適合太高
1.8元空間設(shè)置大小
-XX:MetaspaceSize 初始空間大小,達(dá)到該值就會(huì)觸發(fā)垃圾收集進(jìn)行類型卸載渔彰,同時(shí)GC會(huì)對(duì)該值進(jìn)行調(diào)整:如果釋放了大量的空間嵌屎,就適當(dāng)降低該值推正;如果釋放了很少的空間,那么在不超過(guò)MaxMetaspaceSize時(shí)宝惰,適當(dāng)提高該值植榕。
-XX:MaxMetaspaceSize 最大空間,默認(rèn)是沒(méi)有限制的尼夺。

代碼演示

1.默認(rèn)值測(cè)試

/**
 *
 *  命令查看默認(rèn)值:
 *  最大內(nèi)存4G
 *  初始內(nèi)存:最大內(nèi)存的1/16
 *  垃圾回收器:Parallel Scavenge收集器
 *
 *  java -XX:+PrintCommandLineFlags -version
 * -XX:InitialHeapSize=268435456 -XX:MaxHeapSize=4294967296 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC
 *
 * 代碼查看默認(rèn)值:
 * 最大內(nèi)存:4G
 * 新生代:老年代= 87360:174784 = 1:2
 * eden:from:to = 8:1:1
 *
 * 最大內(nèi)存3959.5M
 * 可用內(nèi)存244.76746368408203M
 * 已經(jīng)使用內(nèi)存247.5M
 * Heap 堆信息
 *  def new generation   total 78656K, used 4197K [0x00000006c0000000, 0x00000006c5550000, 0x0000000715550000)
 *   eden space 69952K,   6% used [0x00000006c0000000, 0x00000006c0419618, 0x00000006c4450000)
 *   from space 8704K,   0% used [0x00000006c4450000, 0x00000006c4450000, 0x00000006c4cd0000)
 *   to   space 8704K,   0% used [0x00000006c4cd0000, 0x00000006c4cd0000, 0x00000006c5550000)
 *  tenured generation   total 174784K, used 0K [0x0000000715550000, 0x0000000720000000, 0x00000007c0000000)
 *    the space 174784K,   0% used [0x0000000715550000, 0x0000000715550000, 0x0000000715550200, 0x0000000720000000)
 *  Metaspace       used 2719K, capacity 4486K, committed 4864K, reserved 1056768K
 *   class space    used 292K, capacity 386K, committed 512K, reserved 1048576K
 *
 */
public class 默認(rèn)值測(cè)試 {
    public static void main(String[] args) {
        System.out.print("最大內(nèi)存");
        System.out.println(Runtime.getRuntime().maxMemory() / 1024.0 / 1024 + "M");
        System.out.print("可用內(nèi)存");
        System.out.println(Runtime.getRuntime().freeMemory() / 1024.0 / 1024 + "M");
        System.out.print("已經(jīng)使用內(nèi)存");
        System.out.println(Runtime.getRuntime().totalMemory() / 1024.0 / 1024 + "M");
    }
}

2.吞吐量測(cè)試

/**
 * -XX:PermSize=32M -Xmx32M -Xms1M -XX:+UseSerialGC -XX:+PrintGCDetails
 * 非堆內(nèi)存32M尊残,堆內(nèi)存最大32M,最小1M淤堵,使用串行垃圾回收器
 * 啟動(dòng)GC-96次 執(zhí)行GC-20次 吞吐量100
 *
 * -XX:PermSize=32M -Xmx32M -Xms32M -XX:+UseSerialGC -XX:+PrintGCDetails
 * 非堆內(nèi)存32M寝衫,堆內(nèi)存最大32M,最小32M拐邪,使用串行垃圾回收器
 * 啟動(dòng)GC-39次 執(zhí)行GC-20次 吞吐量130
 *
 * -XX:PermSize=32M -Xmx512M -Xms32M -XX:+UseSerialGC -XX:+PrintGCDetails
 * 非堆內(nèi)存32M慰毅,堆內(nèi)存最大512M,最小32M庙睡,使用串行垃圾回收器
 * 啟動(dòng)GC-39次 執(zhí)行GC-19次 吞吐量130
 *
 * -XX:PermSize=32M -Xmx512M -Xms512M -XX:+UseSerialGC -XX:+PrintGCDetails
 * 非堆內(nèi)存32M事富,堆內(nèi)存最大512M,最小512M乘陪,使用串行垃圾回收器
 * 啟動(dòng)GC-1次 執(zhí)行GC-1次 吞吐量140
 *
 * ===================結(jié)論:===================================
 * =======================1统台、垃圾回收次數(shù)和最大堆內(nèi)存大小無(wú)關(guān),只和初始內(nèi)存有關(guān)系啡邑。============
 * =======================2贱勃、最大堆內(nèi)存和初始堆內(nèi)存設(shè)置的一樣,可以減少垃圾回收次數(shù)============
 * =======================3谤逼、初始堆內(nèi)存會(huì)影響到吞吐量贵扰,兩個(gè)值都越大吞吐量就越高============
 * =======================4、垃圾回收次數(shù)越少流部,吞吐量越高============
 */
@SpringBootApplication
@RestController
public class 吞吐量測(cè)試 {
    public static void main(String[] args) {
        SpringApplication.run(吞吐量測(cè)試.class);
    }

    @RequestMapping("/test2")
    public void test(){
    }
}

3.垃圾回收器測(cè)試

/**
 * -XX:PermSize=32M -Xmx64M -Xms64M -XX:+UseSerialGC -XX:+PrintGCDetails
 * 啟動(dòng)GC-19次 執(zhí)行GC-10次 吞吐量95
 *
 * -XX:PermSize=32M -Xmx64M -Xms64M -XX:+UseParNewGC -XX:+PrintGCDetails
 * 啟動(dòng)GC-19次 執(zhí)行GC-10次 吞吐量100
 *
 * -XX:PermSize=32M -Xmx64M -Xms64M -XX:+UseConcMarkSweepGC -XX:+PrintGCDetails
 * 啟動(dòng)GC-20次 執(zhí)行GC-10次 吞吐量103
 *
 * -XX:PermSize=32M -Xmx64M -Xms64M -XX:+UseG1GC -XX:+PrintGCDetails
 * 啟動(dòng)GC-17次 執(zhí)行GC-8次 吞吐量102
 *
 * ===================結(jié)論:===================================
 * =======================1戚绕、最好使用并行收集器,因?yàn)椴⑿惺占魉俣缺却型掏铝扛撸俣瓤臁?===========
 */
@SpringBootApplication
@RestController
public class 垃圾回收器測(cè)試 {
    public static void main(String[] args) {
        SpringApplication.run(垃圾回收器測(cè)試.class);
    }

    @RequestMapping("/test")
    public void test(){
    }
}

相關(guān)文章鏈接:
<<<JVM整體內(nèi)存結(jié)構(gòu)的圖解枝冀,直觀明了
<<<javap命令查看對(duì)象信息及操作方法在JVM層的實(shí)現(xiàn)原理
<<<javap命令反查匯編指令匯總
<<<ClassLoader類加載器順序Demo測(cè)試與雙親委派源碼解讀
<<<自定義SPI和熱部署技術(shù)破壞類加載器的雙親委派模式
<<<JVM中對(duì)象如何完成空間分配和初始化工作
<<<JVM元空間(方法區(qū))和棧內(nèi)存溢出原因及解決方案
<<<JVM堆內(nèi)存溢出和內(nèi)存泄露問(wèn)題定位和解決
<<<JVM常見(jiàn)死鎖問(wèn)題產(chǎn)生原因和多種診斷方式
<<<服務(wù)器CPU飆升為100%問(wèn)題排查及如何避免
<<<JVM內(nèi)存診斷命令和排查工具匯總
<<<JVM新生代老年代算法匯總圖解
<<<JVM垃圾回收不要手動(dòng)System.gc的真正原因
<<<JVM垃圾回收引用計(jì)數(shù)法和根搜索算法圖解
<<<JVM垃圾回收STW(Stop-The-World)代碼演示
<<<JVM垃圾回收器的發(fā)展歷程及使用場(chǎng)景匯總
<<<JVM串行并行垃圾回收器的關(guān)注點(diǎn)
<<<一張圖看懂CMS垃圾回收器的底層原理
<<<G1能作為JDK9默認(rèn)垃圾回收器的優(yōu)勢(shì)分析
<<<CMS和G1的漏標(biāo)問(wèn)題解決及三色標(biāo)記算法圖解
<<<GC中新生代進(jìn)入老年代的方式匯總
<<<GC常用日志參數(shù)配置及分析工具說(shuō)明
<<<FullGC舞丛、MinorGC、STW等常見(jiàn)問(wèn)題如何解答

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末果漾,一起剝皮案震驚了整個(gè)濱河市球切,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌绒障,老刑警劉巖吨凑,帶你破解...
    沈念sama閱讀 206,723評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異户辱,居然都是意外死亡鸵钝,警方通過(guò)查閱死者的電腦和手機(jī)糙臼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蒋伦,“玉大人弓摘,你說(shuō)我怎么就攤上這事焚鹊『劢欤” “怎么了?”我有些...
    開封第一講書人閱讀 152,998評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵末患,是天一觀的道長(zhǎng)研叫。 經(jīng)常有香客問(wèn)我,道長(zhǎng)璧针,這世上最難降的妖魔是什么嚷炉? 我笑而不...
    開封第一講書人閱讀 55,323評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮探橱,結(jié)果婚禮上申屹,老公的妹妹穿的比我還像新娘。我一直安慰自己隧膏,他們只是感情好哗讥,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,355評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著胞枕,像睡著了一般杆煞。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上腐泻,一...
    開封第一講書人閱讀 49,079評(píng)論 1 285
  • 那天决乎,我揣著相機(jī)與錄音,去河邊找鬼派桩。 笑死构诚,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的铆惑。 我是一名探鬼主播范嘱,決...
    沈念sama閱讀 38,389評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼鸭津!你這毒婦竟也來(lái)了彤侍?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,019評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤逆趋,失蹤者是張志新(化名)和其女友劉穎盏阶,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體闻书,經(jīng)...
    沈念sama閱讀 43,519評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡名斟,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,971評(píng)論 2 325
  • 正文 我和宋清朗相戀三年脑慧,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片砰盐。...
    茶點(diǎn)故事閱讀 38,100評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡闷袒,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出岩梳,到底是詐尸還是另有隱情囊骤,我是刑警寧澤,帶...
    沈念sama閱讀 33,738評(píng)論 4 324
  • 正文 年R本政府宣布冀值,位于F島的核電站也物,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏列疗。R本人自食惡果不足惜滑蚯,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,293評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望抵栈。 院中可真熱鬧告材,春花似錦、人聲如沸古劲。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)绢慢。三九已至灿渴,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間胰舆,已是汗流浹背骚露。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留缚窿,地道東北人棘幸。 一個(gè)月前我還...
    沈念sama閱讀 45,547評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像倦零,于是被迫代替她去往敵國(guó)和親误续。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,834評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容