無論是YGC仍是Full GC,GC過程當中都會對致使程序運行中中斷,正確的選擇不一樣的GC策略,調整JVM、GC的參數,能夠極大的減小因爲GC工做,而致使的程序運行中斷方面的問題,進而適當的提升Java程序的工做效率。可是調整GC是以個極爲複雜的過程,因爲各個程序具有不一樣的特色,如:web和GUI程序就有很大區別(Web能夠適當的停頓,但GUI停頓是客戶沒法接受的),並且因爲跑在各個機器上的配置不一樣(主要cup個數,內存不一樣),因此使用的GC種類也會不一樣(如何選擇見GC種類及如何選擇)。本文將注重介紹JVM、GC的一些重要參數的設置來提升系統的性能。html
JVM內存組成及GC相關內容請見以前的文章:JVM內存組成 GC策略&內存申請。java
JVM參數的含義 實例見實例分析web
參數名稱 | 含義 | 默認值 | |
-Xms | 初始堆大小 | 物理內存的1/64(<1GB) | 默認(MinHeapFreeRatio參數能夠調整)空餘堆內存小於40%時,JVM就會增大堆直到-Xmx的最大限制. |
-Xmx | 最大堆大小 | 物理內存的1/4(<1GB) | 默認(MaxHeapFreeRatio參數能夠調整)空餘堆內存大於70%時,JVM會減小堆直到 -Xms的最小限制 |
-Xmn | 年輕代大小(1.4or lator) | 注意: 整個堆大小=年輕代大小 + 年老代大小 增大年輕代後,將會減少年老代大小.此值對系統性能影響較大,Sun官方推薦配置爲整個堆的3/8 |
|
-XX:NewSize | 設置年輕代大小(for 1.3/1.4) | ||
-XX:MaxNewSize | 年輕代最大值(for 1.3/1.4) | ||
-XX:PermSize | 設置持久代(perm gen)初始值 | 物理內存的1/64 | |
-XX:MaxPermSize | 設置持久代最大值 | 物理內存的1/4 | |
-Xss | 每一個線程的堆棧大小 | JDK5.0之後每一個線程堆棧大小爲1M,之前每一個線程堆棧大小爲256K.更具應用的線程所需內存大小進行 調整.在相同物理內存下,減少這個值能生成更多的線程.可是操做系統對一個進程內的線程數仍是有限制的,不能無限生成,經驗值在3000~5000左右 通常小的應用, 若是棧不是很深, 應該是128k夠用的 大的應用建議使用256k。這個選項對性能影響比較大,須要嚴格的測試。(校長) 和threadstacksize選項解釋很相似,官方文檔彷佛沒有解釋,在論壇中有這樣一句話:"」 -Xss is translated in a VM flag named ThreadStackSize」 通常設置這個值就能夠了。 |
|
-XX:ThreadStackSize | Thread Stack Size | (0 means use default stack size) [Sparc: 512; Solaris x86: 320 (was 256 prior in 5.0 and earlier); Sparc 64 bit: 1024; Linux amd64: 1024 (was 0 in 5.0 and earlier); all others 0.] | |
-XX:NewRatio | 年輕代(包括Eden和兩個Survivor區)與年老代的比值(除去持久代) | -XX:NewRatio=4表示年輕代與年老代所佔比值爲1:4,年輕代佔整個堆棧的1/5 Xms=Xmx而且設置了Xmn的狀況下,該參數不須要進行設置。 |
|
-XX:SurvivorRatio | Eden區與Survivor區的大小比值 | 設置爲8,則兩個Survivor區與一個Eden區的比值爲2:8,一個Survivor區佔整個年輕代的1/10 | |
-XX:LargePageSizeInBytes | 內存頁的大小不可設置過大, 會影響Perm的大小 | =128m | |
-XX:+UseFastAccessorMethods | 原始類型的快速優化 | ||
-XX:+DisableExplicitGC | 關閉System.gc() | 這個參數須要嚴格的測試 | |
-XX:MaxTenuringThreshold | 垃圾最大年齡 | 若是設置爲0的話,則年輕代對象不通過Survivor區,直接進入年老代. 對於年老代比較多的應用,能夠提升效率.若是將此值設置爲一個較大值,則年輕代對象會在Survivor區進行屢次複製,這樣能夠增長對象再年輕代的存活 時間,增長在年輕代即被回收的機率 該參數只有在串行GC時纔有效. |
|
-XX:+AggressiveOpts | 加快編譯 | ||
-XX:+UseBiasedLocking | 鎖機制的性能改善 | ||
-Xnoclassgc | 禁用垃圾回收 | ||
-XX:SoftRefLRUPolicyMSPerMB | 每兆堆空閒空間中SoftReference的存活時間 | 1s | softly reachable objects will remain alive for some amount of time after the last time they were referenced. The default value is one second of lifetime per free megabyte in the heap |
-XX:PretenureSizeThreshold | 對象超過多大是直接在舊生代分配 | 0 | 單位字節 新生代採用Parallel Scavenge GC時無效 另外一種直接在舊生代分配的狀況是大的數組對象,且數組中無外部引用對象. |
-XX:TLABWasteTargetPercent | TLAB佔eden區的百分比 | 1% | |
-XX:+CollectGen0First | FullGC時是否先YGC | false |
GC性能方面的考慮算法
對於GC的性能主要有2個方面的指標:吞吐量throughput(工做時間不算gc的時間佔總的時間比)和暫停pause(gc發生時app對外顯示的沒法響應)。spring
1. Total Heap數組
默認狀況下,vm會增長/減小heap大小以維持free space在整個vm中佔的比例,這個比例由MinHeapFreeRatio和MaxHeapFreeRatio指定。緩存
通常而言,server端的app會有如下規則:併發
2. The Young Generationapp
另一個對於app流暢性運行影響的因素是young generation的大小。young generation越大,minor collection越少;可是在固定heap size狀況下,更大的young generation就意味着小的tenured generation,就意味着更多的major collection(major collection會引起minor collection)。框架
NewRatio反映的是young和tenured generation的大小比例。NewSize和MaxNewSize反映的是young generation大小的下限和上限,將這兩個值設爲同樣就固定了young generation的大小(同Xms和Xmx設爲同樣)。
若是但願,SurvivorRatio也能夠優化survivor的大小,不過這對於性能的影響不是很大。SurvivorRatio是eden和survior大小比例。
通常而言,server端的app會有如下規則:
經驗&&規則
java.lang.StackOverflowError:(不多)
java.lang.OutOfMemoryError:heap space(比較常見)
java.lang.OutOfMemoryError: PermGen space (常常出現)
java.lang.OutOfMemoryError: PermGen space 這個異常比較常見,是說JVM裏的Perm內存區的異常溢出,因爲JVM在默認的狀況下,Perm默認爲64M,而不少程序須要大量的Perm區內存,尤爲使用到像spring等框架的時候,因爲須要使用到動態生成類,而這些類不能被GC自動釋放,因此致使OutOfMemoryError: PermGen space異常。解決方法很簡單,增大JVM的 -XX:MaxPermSize 啓動參數,就能夠解決這個問題
java.lang.OutOfMemoryError:heap space或 其它OutOfMemoryError,這個異常實際上跟上面的異常是一個異常,但解決方法不一樣,因此分開來寫。上面那個異常是由於JVM的perm區內存區分少了引發的(JVM的內存區分爲 young,old,perm三種)。而這個異常是由於JVM堆內存或者說整體分少了。解決方法是更改 -Xms -Xmx 啓動參數,一般是擴大1倍。xms是管理啓動時最小內存量的,xmx是管裏JVM最大的內存量的
在單線程操做中,不管是棧深度無限增長,仍是棧幀(每一個方法調用執行時都會在棧中建立一個棧幀,用來存儲局部變量,操做數棧,動態鏈表,方法出口等信息)佔的空間太大,都出現的是StackOverflowError