1.1. 堆大小設置
JVM 中最大堆大小有三方面限制: n 相關操作系統的數據模型(32-bt還是64-bit)限制; n 系統的可用虛擬內存限制; n 系統的可用物理內存限制。 32位系統下,一般限制在1.5G~2G;64為操作系統對內存無限制。 1.2. 典型設置:
java -Xmx3550m -Xms3550m -Xmn2g-Xss128k 所以增大年輕代後,將會減小年老代大小。此值對系統性能影響較大,Sun官方推薦配置為整個堆的3/8。 java -Xmx3550m -Xms3550m -Xss128k -XX:NewRatio=4 -XX:SurvivorRatio=4 -XX:MaxPermSize=16m -XX:MaxTenuringThreshold=0 1.3. 回收器選擇
JVM給了三種選擇:串行收集器、並行收集器、並發收集器,但是串行收集器只適用於小數據量的情況,所以這裏的選擇主要針對並行收集器和並發收集器。默認情況下,JDK5.0以前都是使用串行收集器,如果想使用其他收集器需要在啟動時加入相應參數。JDK5.0以後,JVM會根據當前系統配置進行判斷。 1.3.1. 吞吐量優先的並行收集器
如上文所述,並行收集器主要以到達一定的吞吐量為目標,適用於科學技術和後臺處理等。 java -Xmx3800m -Xms3800m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:ParallelGCThreads=20 -XX:ParallelGCThreads=20:配置並行收集器的線程數,即:同時多少個線程一起進行垃圾回收。此值最好配置與處理器數目相等。 java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:ParallelGCThreads=20-XX:+UseParallelOldGC-XX:+UseParallelOldGC: 配置年老代垃圾收集方式為並行收集。JDK6.0支持對年老代並行收集。 java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:MaxGCPauseMillis=100-XX:MaxGCPauseMillis=100:設置每次年輕代垃圾回收的最長時間,如果無法滿足此時間,JVM會自動調整年輕代大小,以滿足此值。 java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:MaxGCPauseMillis=100-XX:+UseAdaptiveSizePolicy 1.3.2. 響應時間優先的並發收集器
如上文所述,並發收集器主要是保證系統的響應時間,減少垃圾收集時的停頓時間。適用於應用服務器、電信領域等。 java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:ParallelGCThreads=20-XX:+UseConcMarkSweepGC -XX:+UseParNewGC-XX:+UseConcMarkSweepGC:設置年老代為並發收集。測試中配置這個以後,-XX:NewRatio=4的配置失效了,原因不明。所以,此時年輕代大小最好用-Xmn設置。 java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseConcMarkSweepGC-XX:CMSFullGCsBeforeCompaction=5-XX:+UseCMSCompactAtFullCollection 輔助信息 JVM提供了大量命令行參數,打印信息,供調試使用。主要有以下一些: -XX:+PrintGC輸出形式: [GC 118250K->113543K(130112K), 0.0094143 secs] [Full GC 121376K->10414K(130112K), 0.0650971 secs] -XX:+PrintGCDetails輸出形式: [GC [DefNew: 8614K->781K(9088K), 0.0123035 secs] 118250K->113543K(130112K), 0.0124633 secs] [GC [DefNew: 8614K->8614K(9088K), 0.0000665 secs][Tenured: 112761K->10414K(121024K), 0.0433488 secs] 121376K->10414K(130112K), 0.0436268 secs] -XX:+PrintGCTimeStamps-XX:+PrintGC:PrintGCTimeStamps可與上面兩個混合使用 -XX:+PrintGCApplicationConcurrentTime:打印每次垃圾回收前,程序未中斷的執行時間。可與上面混合使用 -XX:+PrintGCApplicationStoppedTime:打印垃圾回收期間程序暫停的時間。可與上面混合使用 -XX:PrintHeapAtGC:打印GC前後的詳細堆棧信息 o -Xloggc:filename:與上面幾個配合使用,把相關日志信息記錄到文件以便分析。 常見配置匯總 1 堆設置 § -Xms:初始堆大小 § -Xmx:最大堆大小 § -XX:NewSize=n:設置年輕代大小 § -XX:NewRatio=n:設置年輕代和年老代的比值。如:為3,表示年輕代與年老代比值為1:3,年輕代占整個年輕代年老代和的1/4 § -XX:SurvivorRatio=n:年輕代中Eden區與兩個Survivor區的比值。注意Survivor區有兩個。如:3,表示Eden:Survivor=3:2,一個Survivor區占整個年輕代的1/5 § -XX:MaxPermSize=n:設置持久代大小 1 收集器設置 § -XX:+UseSerialGC:設置串行收集器 § -XX:+UseParallelGC:設置並行收集器 § -XX:+UseParalledlOldGC:設置並行年老代收集器 § -XX:+UseConcMarkSweepGC:設置並發收集器 1 垃圾回收統計信息 § -XX:+PrintGC § -XX:+PrintGCDetails § -XX:+PrintGCTimeStamps § -Xloggc:filename 1 並行收集器設置 § -XX:ParallelGCThreads=n:設置並行收集器收集時使用的CPU數。並行收集線程數。 § -XX:MaxGCPauseMillis=n:設置並行收集最大暫停時間 § -XX:GCTimeRatio=n:設置垃圾回收時間占程序運行時間的百分比。公式為1/(1+n) 1 並發收集器設置 § -XX:+CMSIncrementalMode:設置為增量模式。適用於單CPU情況。 -XX:ParallelGCThreads=n:設置並發收集器年輕代收集方式為並行收集時,使用的CPU數。並行收集線程數。 1.4. 調優總結
1.4.1. 年輕代大小選擇
響應時間優先的應用:盡可能設大,直到接近系統的最低響應時間限制(根據實際情況選擇)。在此種情況下,年輕代收集發生的頻率也是最小的。同時,減少到達年老代的對象。 吞吐量優先的應用:盡可能的設置大,可能到達Gbit的程度。因為對響應時間沒有要求,垃圾收集可以並行進行,一般適合8CPU以上的應用。 1.4.2. 年老代大小選擇
響應時間優先的應用: 年老代使用並發收集器,所以其大小需要小心設置,一般要考慮並發會話率和會話持續時間等一些參數。如果堆設置小了,可以會造成內存碎片、高回收頻率以及應用暫停而使用傳統的標記清除方式;如果堆大了,則需要較長的收集時間。最優化的方案,一般需要參考以下數據獲得: n 並發垃圾收集信息 n 持久代並發收集次數 n 傳統GC信息 n 花在年輕代和年老代回收上的時間比例 減少年輕代和年老代花費的時間,一般會提高應用的效率 吞吐量優先的應用: 一般吞吐量優先的應用都有一個很大的年輕代和一個較小的年老代。原因是,這樣可以盡可能回收掉大部分短期對象,減少中期的對象,而年老代盡存放長期存活對象。 較小堆引起的碎片問題: 因為年老代的並發收集器使用標記、清除算法,所以不會對堆進行壓縮。當收集器回收時,他會把相鄰的空間進行合並,這樣可以分配給較大的對象。但是,當堆空間較小時,運行一段時間以後,就會出現“碎片”,如果並發收集器找不到足夠的空間,那麼並發收集器將會停止,然後使用傳統的標記、清除方式進行回收。如果出現“碎片”,可能需要進行如下配置: -XX:+UseCMSCompactAtFullCollection:使用並發收集器時,開啟對年老代的壓縮。 -XX:CMSFullGCsBeforeCompaction=0:上面配置開啟的情況下,這裏設置多少次Full GC後,對年老代進行壓縮 2. Resin的調優
要綜合考慮resin線程池大小、“-Xmx:JVM最大可用內存”、“-Xms:初始堆大小”、“-Xmn:Young Generation的heap size”參數互相匹配。(JVM有2個GC線程。第一個線程負責回收Heap的Young區。第二個線程在Heap不足時,遍曆Heap,將Young區升級為Older區。Older區的大小等於-Xmx減去-Xmn,不能將-Xms的值設的過大,因為第二個線程被迫運行會降低JVM的性能。) 來避免線上resin服務反複出現以下異常: l OutOfMemoryError: Java heap space l OutOfMemoryError: PermGen space 2.1. 在Resin.xml中設置JDK參數:
Resin 4.0已支持把JDK參數加入resin配置文件resin.xml裏。 參考resin的幫助文檔: 『JDK arguments Resin 4.0 has moved all JDK arguments into the resin.xml file, in the <jvm-arg> tag. Because the Resin 4.0 watchdog starts each Resin server instance, it can pass the arguments defined in the configuration file to the JVM. By moving the Java arguments to the configuration file, server configuration is easier and more maintainable.』 2.2. 建議規則:
1、 Server端JVM最好將-Xms和-Xmx設為相同值。為了優化GC,最好讓-Xmn值約等於-Xmx的1/4。 2、 通過增大“-XX:PermSize”和“-XX:MaxPermSize”這兩個參數來避免出現JVM內存永久保存區域溢出引發Resin的500錯誤。(因為spring+struts的流行,這些框架用到大量動態class,ClassLoader是把這部分內存放在PermGen space裏的。而JVM的GC是不會清理PermGen space的。這樣容易導致線上應用報告PermGen space內存溢出。) 4、建議resin配置: 建議生產環境下部署Resin 4的resin.xml中增加如下配置節點: <server-default> <jvm-arg>-Xms1024m</jvm-arg> <jvm-arg>-Xmx1024m</jvm-arg> <jvm-arg>-Xmn256m</jvm-arg> <jvm-arg>-XX:PermSize=128m</jvm-arg> <jvm-arg>-XX:MaxPermSize=256m</jvm-arg> <thread-max>1024</thread-max> <socket-timeout>30s</socket-timeout> <keepalive-max>512</keepalive-max> <keepalive-timeout>60s</keepalive-timeout> </server-default> |
|