JVM 调优(学习篇)
概述
公司的江南白衣寫了一篇關鍵業務系統的JVM參數推薦(2016熱冬版)的文章,大牛的文章總是需要細細品讀。這篇文章介紹大量的JVM調優參數,內容也比較多,本文只是列出我自己能理解的一些參數,暫時理解不了的參數就只能等以后自己實力到家了,再慢慢補充上來。
?
性能調優參數
-XX:AutoBoxCacheMax
JAVA進程啟動的時候,會加載rt.jar這個核心包,rt.jar包里的Integer自然也會被加載到JVM中,Integer里面有一個IntegerCache緩存,如下:
private?static?class?IntegerCache?{static?final?int?low?=?-128;static?final?int?high;static?final?Integer?cache[];static?{//?high?value?may?be?configured?by?propertyint?h?=?127;String?integerCacheHighPropValue?=sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");if?(integerCacheHighPropValue?!=?null)?{int?i?=?parseInt(integerCacheHighPropValue);i?=?Math.max(i,?127);//?Maximum?array?size?is?Integer.MAX_VALUEh?=?Math.min(i,?Integer.MAX_VALUE?-?(-low)?-1);}high?=?h;cache?=?new?Integer[(high?-?low)?+?1];int?j?=?low;for(int?k?=?0;?k?<?cache.length;?k++)cache[k]?=?new?Integer(j++);}private?IntegerCache()?{} }IntegerCache有一個靜態代碼塊,JVM在加載Integer這個類時,會優先加載靜態的代碼。當JVM進程啟動完畢后, -128 ~ +127 范圍的數字會被緩存起來,調用valueOf方法的時候,如果是這個范圍內的數字,則直接從緩存取出。
超過這個范圍的,就只能構造新的Integer對象了。
因此可以根據實際情況把AutoBoxCacheMax的值設置的大些,比如江南白衣推薦的
-XX:AutoBoxCacheMax=20000?
-XX:+AlwaysPreTouch
JAVA進程啟動的時候,雖然我們可以為JVM指定合適的內存大小,但是這些內存操作系統并沒有真正的分配給JVM,而是等JVM訪問這些內存的時候,才真正分配,這樣會造成以下問題。
1、GC的時候,新生代的對象要晉升到老年代的時候,需要內存,這個時候操作系統才真正分配內存,這樣就會加大young gc的停頓時間;
2、可能存在內存碎片的問題。
可以在JVM啟動的時候,配置
-XX:+AlwaysPreTouch參數,這樣JVM就會先訪問所有分配給它的內存,讓操作系統把內存真正的分配給JVM,后續JVM就可以順暢的訪問內存了。
?
GC參數
JAVA 1.7用的垃圾收集算法還是CMS,下文提到的參數都是針對CMS的。
CMSInitiatingOccupancyFraction
之前寫過一篇java垃圾回收算法之-CMS(并發標記清除),里面提到垃圾收集線程會跟應用的線程一起并行的工作,萬一垃圾收集線程在工作的時候,老年代內存不足怎么辦?因此最好還是提前啟動CMS來收集垃圾(CMS GC)。
可以通過設置
那么當老年代堆空間的使用率達到75%的時候就開始執行垃圾回收,CMSInitiatingOccupancyFraction默認值是92%,這個就太大了。
CMSInitiatingOccupancyFraction參數必須跟下面兩個參數一起使用才能生效的。
?
MaxTenuringThreshold
新生代是使用copy算法來進行垃圾回收的,可以參看
java垃圾回收算法之-coping復制
默認情況下,當新生代執行了15次young gc后,如果還有對象存活在Survivor區中,那么就可以直接將這些對象晉升到老年代,但是由于新生代使用copy算法,如果Survivor區存活的對象太久的話,Survivor區存活的對象就越多,這個就會影響copy算法的性能,使得young gc停頓的時間加長,建議設置成6。
-XX:MaxTenuringThreshold=6?
ExplicitGCInvokesConcurrent
如果系統使用堆外內存,比如用到了Netty的DirectByteBuffer類,那么當想回收堆外內存的時候,需要調用
System.gc()而這個方法將進行full gc,整個應用將會停頓,如果是使用CMS垃圾收集器,那么可以設置
-XX:+ExplicitGCInvokesConcurrent這個參數來改變System.gc()的行為,讓其從full gc --> CMS GC,CMS GC是并發收集的,且中間執行的過程中,只有部分階段需要stop the world。
注意:設置了ExplicitGCInvokesConcurrent,那就不要設置DisableExplicitGC參數來禁掉System.gc()。
?
內存參數
-Xmx, -Xms
這兩個一般都是設置4個g
?
NewRatio
GC最多的還是發生在新生代的young gc,所以可以提高一下新生代在整個堆的占用比例,建議設置為對半分,盡量避免young gc
-XX:NewRatio=1總結
以上是生活随笔為你收集整理的JVM 调优(学习篇)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: MySQL InnoDB 锁介绍及不同
- 下一篇: Kafka 压测:3 台廉价服务器竟支撑