原帖:http://www.javaeye.com/topic/756538html
性能優化從身邊作起。
首先創建評估體系,將workspace裏全部的項目close掉,關閉eclipse。優化的用例就是啓動eclipse,open一個項目,eclipse會自動build這個項目,保證沒有感受到明顯的卡,也就是沒有full GC。
開始:
eclipse.ini里加入打印gc狀況的參數:
java
-XX:+PrintGCTimeStamps |
-XX:+PrintGCDetails |
-verbose:gc |
-Xloggc:gc.log |
這樣eclipse在運行過程當中會記錄gc日誌,顯示詳細的gc狀況,並打印在gc.log中,經過分析這個日誌尋找eclipse的性能瓶頸和優化方式。
我最初的參數只是在原版基礎上調了堆大小 數組
-Xms512m |
-Xmx512m |
將堆初始化和最大值設爲同樣,消除堆大小根據當前堆使用狀況而變化帶來的影響。
啓動eclipse,發現gc.log裏打出了不少full gc的日誌 緩存
在啓動的6秒多時間裏共出現了8次full gc,因此啓動慢,以爲啓動時候挺卡的。從日誌裏能夠看出來 FullGC主要是在回收tenured區和Perm區,其中Perm一直都是快滿的狀態,Perm : 24574K->24554K(24576K),Perm大小在不斷調整,因此須要固定Perm區的大小,保證夠用,eclipse.ini里加入 性能優化
-XX:PermSize=64m |
-XX:MaxPermSize=64m |
再啓動:發現沒有full gc了只有數量比較多的minor gc,挑啓動開始到啓動完成的第一條和最後一條日誌 服務器
這6秒中GC日誌打了69次, 而內存回收率仍是蠻高的 young區18880-1985=16895 jvm 46992-30098=16894 都快接近100%了,能夠看出young區是由小到大在不斷調整大小,因此不斷GC,所以設一個初始值吧,聽說設置heap的1/4比較好,那就是 128M,因此eclipse.ini加入 eclipse
-Xmn128m |
再重啓,發現GC日誌就四條了,eclipse啓動天然快了 jvm
可是,啓動後open個人多個項目,這些項目互相依賴,eclipse自動build,感受有點小卡,發現日誌裏多了4次full GC,因此就卡了… ide
這個時候Tenured區和Perm都還沒到很接近最大值,可是爲何還有full GC呢,開始覺得是JVM悲觀認爲Tenured區剩餘空間不足以應對下一次minor GC 因此進行了full GC調整Tenured空間,索性直接增長了堆最大值到-Xmx728m(工做電腦的內存是3.5G),但重啓後full gc仍是有4次,並且有幾回minor GC用的時間超過了0.1秒,這是由於增長了堆大小,致使GC用時也增長了,不能接受。因此仍是改回-Xmx512m。
再仔細觀察日誌,發現Full GC (System) 字樣,這個意思是eclipse裏調用了System.gc()手動觸發了系統GC,好吧,哥已經給你分配足夠空間了,你就省省吧,在eclipse.ini里加入: 工具
-XX:+DisableExplicitGC |
這樣就差很少了,整個過程沒有出現full gc,再編碼2個小時,中間只出現了一次full gc,在open build某50W行+的代碼的時候,eclipse仍是卡了…
最後又稍微調了一下各代的大小,獲得目前的參數:
-Xms512m |
-Xmx512m |
-XX:PermSize=96m |
-XX:MaxPermSize=96m |
-Xmn168m |
-XX:+DisableExplicitGC |
另外沒有去調GC策略,主要是以爲eclipse是客戶端程序,默認的client單線程的GC策略應該是比較適合的,之後有時間再試試看吧。
Java 虛擬機默認分配64M內存,若是你的應用比較大,超出64M內 存,Java虛擬機就會拋出outOfMemoryError,並中止運行。無論是什麼應用(Web應用、Application等),只須要修改你的機 器上的運行Java命令,在javaxxx命令中添加-Xms(最小使用內存)、-Xmx(最大使用內存)便可解決。固然,這兒的內存容量都是指物理內 存,不能超出你的機器的物理內存的總容量。
對於單獨的.class,能夠用下面的方法對Test運行時的jvm內存進行設置。
java -Xms64m -Xmx256m Test
-Xms是設置內存初始化的大小
-Xmx是設置最大可以使用內存的大小(最好不要超過物理內存大小)
在eclipse中的設置方法
方 法1:對應在Eclipse中的設置爲:窗口->首選項->JAVA->已安裝的JRE,在缺 省的VM自變量中增長:-Xmx256M (也能夠添加其餘參數如-Xms128M -XX:PermSize=64M -XX:MaxPermSize=128M)
方法2:修改eclipse.ini
-vmargs -Xms128M -Xmx512M -XX:PermSize=64M -XX:MaxPermSize=128M
這裏有幾個問題:
1. 各個參數的含義什麼?
2. 爲何有的機器我將-Xmx和-XX:MaxPermSize都設置爲512M以後Eclipse能夠啓動,而有些機器沒法啓動?
3. 爲什麼將上面的參數寫入到eclipse.ini文件Eclipse沒有執行對應的設置?
下面咱們一一進行回答
1. 各個參數的含義什麼?
參數中-vmargs的意思是設置JVM參數,因此後面的其實都是JVM的參數了,咱們首先了解一下JVM內存管理的機制,而後再解釋每一個參數表明的含義。
堆(Heap)和非堆(Non-heap)內存
按 照官方的說法:「Java 虛擬機具備一個堆,堆是運行時數據區域,全部類實例和數組的內存均今後處分配。堆是在 Java 虛擬機啓動時建立的。」「在JVM中堆以外的內存稱爲非堆內存(Non-heap memory)」。能夠看出JVM主要管理兩種類型的內存:堆和非堆。簡單來講堆就是Java代碼可及的內存,是留給開發人員使用的;非堆就是JVM留給 本身用的,因此方法區、JVM內部處理或優化所需的內存(如JIT編譯後的代碼緩存)、每一個類結構(如運行時常數池、字段和方法數據)以及方法和構造方法 的代碼都在非堆內存中。
堆內存分配
JVM初始分配的內存由-Xms指定,默認是物理內存的1/64;JVM最大分配的內存由-Xmx指 定,默認是物理內存的1/4。默認空餘堆內存小於 40%時,JVM就會增大堆直到-Xmx的最大限制;空餘堆內存大於70%時,JVM會減小堆直到-Xms的最小限制。所以服務器通常設置-Xms、 -Xmx相等以免在每次GC 後調整堆的大小。
非堆內存分配
JVM使用-XX:PermSize設置非堆內存初始值,默認是物理內存的1/64;由XX:MaxPermSize設置最大非堆內存的大小,默認是物理內存的1/4。
JVM內存限制(最大值)
首 先JVM內存限制於實際的最大物理內存(廢話!呵呵),假設物理內存無限大的話,JVM內存的最大值跟操做系統有很大的關係。簡單的說就32位處理器雖然 可控內存空間有4GB,可是具體的操做系統會給一個限制,這個限制通常是2GB-3GB(通常來講Windows系統下爲1.5G-2G,Linux系統 下爲2G-3G),而64bit以上的處理器就不會有限制了。
2. 爲何有的機器我將-Xmx和-XX:MaxPermSize都設置爲512M以後Eclipse能夠啓動,而有些機器沒法啓動?
經過上面對JVM內存管理的介紹咱們已經瞭解到JVM內存包含兩種:堆內存和非堆內存,另外JVM最大內存首先取決於實際的物理內存和操做系統。因此說設置VM參數致使程序沒法啓動主要有如下幾種緣由:
1) 參數中-Xms的值大於-Xmx,或者-XX:PermSize的值大於-XX:MaxPermSize;
2) -Xmx的值和-XX:MaxPermSize的總和超過了JVM內存的最大限制,好比當前操做系統最大內存限制,或者實際的物理內存等等。說到實際物理 內存這裏須要說明一點的是,若是你的內存是1024MB,但實際系統中用到的並不多是1024MB,由於有一部分被硬件佔用了。
3. 爲什麼將上面的參數寫入到eclipse.ini文件Eclipse沒有執行對應的設置?
那爲何一樣的參數在快捷方式或者命令行中有效而在eclipse.ini文件中是無效的呢?這是由於咱們沒有遵照eclipse.ini文件的設置規則:
參數形如「項 值」這種形式,中間有空格的須要換行書寫,若是值中有空格的須要用雙引號包括起來。好比咱們使用-vm C:\Java\jre1.6.0\bin\javaw.exe參數設置虛擬機,在eclipse.ini文件中要寫成這樣:
-vm
C:\Java\jre1.6.0\bin\javaw.exe
按照上面所說的,最後參數在eclipse.ini中能夠寫成這個樣子:
-vmargs
-Xms128M
-Xmx512M
-XX:PermSize=64M
-XX:MaxPermSize=128M
實際運行的結果能夠經過Eclipse中「Help」-「About Eclipse SDK」窗口裏面的「Configuration Details」按鈕進行查看。
另外一篇:一次Java垃圾收集調優實戰
GC調優是個很實驗很伽利略的活兒,GC日誌是先決的數據參考和最終驗證:
-XX:+PrintGCDetails -XX:+PrintGCTimeStamps(GC發生的時間) -XX:+PrintGCApplicationStoppedTime(GC消耗了多少時間) -XX:+PrintGCApplicationConcurrentTime(GC之間運行了多少時間)
配置參數:-XX:+UseConcMarkSweepGC
已默認無需配置的參數:-XX:+UseParNewGC(Parallel收集新生代) -XX:+CMSPermGenSweepingEnabled(CMS收集持久代) -XX:UseCMSCompactAtFullCollection(full gc時壓縮年老代)
初始效果:1g堆內存的新生代約60m,minor gc約5-20毫秒,full gc約130毫秒。
配置參數: -XX:+UseParallelGC -XX:+UseParallelOldGC(Parallel收集年老代,從JDK6.0開始支持)
已默認無需配置的參數: -XX:+UseAdaptiveSizePolicy(動態調整新生代大小)
初始效果:1g堆內存的新生代約90-110m(動態調整),minor gc約5-20毫秒,full gc有無UseParallelOldGC 參數分別爲1.3/1.1秒,差異不大。
另外-XX:MaxGCPauseMillis=100 設置minor gc的指望最大時間,JVM會以此來調整新生代的大小,但在此測試環境中對象死的太快,此參數做用不大。
Parallel收集高達1秒的暫停時間基本不可忍受,因此選擇CMS收集器。
在被壓測的Mule 2.0應用裏,每秒都有大約400M的海量短命對象產生:
對這兩個參數的調優,既要改善上面兩種狀況,又要避免新生代過大,複製次數過多形成minor gc的暫停時間過長。
優化後,大約1.1秒才發生一次minor gc,且速度依然保持在15-20ms之間。同時年老代的增加速度大大減緩,好久才發生一次full gc,
參數定稿:
-server -Xms1024m -Xmx1024m -Xmn500m -XX:+UseConcMarkSweepGC -XX:MaxTenuringThreshold=5 -XX:+ExplicitGCInvokesConcurrent
最後服務處理速度從1180 tps 上升到1380 tps,調整兩個參數提高17%的性能仍是筆很划算的買賣。
另外,JDK6 Update 7自帶了一個VisualVM工具,內裏就是以前也有用過的Netbean Profiler,相似JConsole同樣使用,能夠看到線程狀態,內存中對象以及方法的CPU時間等調優重要參考依據。免費捆綁啊,Sun 這樣搞法,其餘作Profiler的公司要關門了。