日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

java虚拟机性能监控调优及原则

發布時間:2024/2/28 编程问答 24 豆豆
生活随笔 收集整理的這篇文章主要介紹了 java虚拟机性能监控调优及原则 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

轉載自?https://www.cnblogs.com/thingk/p/6840585.html

摘抄 http://uule.iteye.com/blog/2114697

?

?

一、JVM內存模型及垃圾收集算法

?1.根據Java虛擬機規范,JVM將內存劃分為:

  • New(年輕代)
  • Tenured(年老代)
  • 永久代(Perm)

? 其中New和Tenured屬于堆內存,堆內存會從JVM啟動參數(-Xmx:3G)指定的內存中分配,Perm不屬于堆內存,有虛擬機直接分配,但可以通過-XX:PermSize -XX:MaxPermSize?等參數調整其大小。

?

  • 年輕代(New):年輕代用來存放JVM剛分配的Java對象
  • 年老代(Tenured):年輕代中經過垃圾回收沒有回收掉的對象將被Copy到年老代
  • 永久代(Perm):永久代存放Class、Method元信息,其大小跟項目的規模、類、方法的量有關,一般設置為128M就足夠,設置原則是預留30%的空間。

New又分為幾個部分:

  • Eden:Eden用來存放JVM剛分配的對象
  • Survivor1
  • Survivro2:兩個Survivor空間一樣大,當Eden中的對象經過垃圾回收沒有被回收掉時,會在兩個Survivor之間來回Copy,當滿足某個條件,比如Copy次數,就會被Copy到Tenured。顯然,Survivor只是增加了對象在年輕代中的逗留時間,增加了被垃圾回收的可能性。

?2.垃圾回收算法

? 垃圾回收算法可以分為三類,都基于標記-清除(復制)算法:

  • Serial算法(單線程)
  • 并行算法
  • 并發算法

? JVM會根據機器的硬件配置對每個內存代選擇適合的回收算法,比如,如果機器多于1個核,會對年輕代選擇并行算法,關于選擇細節請參考JVM調優文檔。

? 稍微解釋下的是,并行算法是用多線程進行垃圾回收,回收期間會暫停程序的執行,而并發算法,也是多線程回收,但期間不停止應用執行。所以,并發算法適用于交互性高的一些程序。經過觀察,并發算法會減少年輕代的大小,其實就是使用了一個大的年老代,這反過來跟并行算法相比吞吐量相對較低。

?

? 還有一個問題是,垃圾回收動作何時執行?

  • 當年輕代內存滿時,會引發一次普通GC,該GC僅回收年輕代。需要強調的時,年輕代滿是指Eden代滿,Survivor滿不會引發GC
  • 當年老代滿時會引發Full GC,Full GC將會同時回收年輕代、年老代
  • 當永久代滿時也會引發Full GC,會導致Class、Method元信息的卸載

? 另一個問題是,何時會拋出OutOfMemoryException,并不是內存被耗空的時候才拋出

  • JVM98%的時間都花費在內存回收
  • 每次回收的內存小于2%

? 滿足這兩個條件將觸發OutOfMemoryException,這將會留給系統一個微小的間隙以做一些Down之前的操作,比如手動打印Heap Dump。

?

二、內存泄漏及解決方法

?1.系統崩潰前的一些現象:

  • 每次垃圾回收的時間越來越長,由之前的10ms延長到50ms左右,FullGC的時間也有之前的0.5s延長到4、5s
  • FullGC的次數越來越多,最頻繁時隔不到1分鐘就進行一次FullGC
  • 年老代的內存越來越大并且每次FullGC后年老代沒有內存被釋放

?之后系統會無法響應新的請求,逐漸到達OutOfMemoryError的臨界值。

?

?2.生成堆的dump文件

?通過JMX的MBean生成當前的Heap信息,大小為一個3G(整個堆的大小)的hprof文件,如果沒有啟動JMX可以通過Java的jmap命令來生成該文件。

?

?3.分析dump文件

?下面要考慮的是如何打開這個3G的堆信息文件,顯然一般的Window系統沒有這么大的內存,必須借助高配置的Linux。當然我們可以借助X-Window把Linux上的圖形導入到Window。我們考慮用下面幾種工具打開該文件:

  • Visual VM
  • IBM HeapAnalyzer
  • JDK 自帶的Hprof工具
  • ?使用這些工具時為了確保加載速度,建議設置最大內存為6G。使用后發現,這些工具都無法直觀地觀察到內存泄漏,Visual VM雖能觀察到對象大小,但看不到調用堆棧;HeapAnalyzer雖然能看到調用堆棧,卻無法正確打開一個3G的文件。因此,我們又選用了Eclipse專門的靜態內存分析工具:Mat。

    ?

    ?4.分析內存泄漏

    ?通過Mat我們能清楚地看到,哪些對象被懷疑為內存泄漏,哪些對象占的空間最大及對象的調用關系。針對本案,在ThreadLocal中有很多的JbpmContext實例,經過調查是JBPM的Context沒有關閉所致。

    ?另,通過Mat或JMX我們還可以分析線程狀態,可以觀察到線程被阻塞在哪個對象上,從而判斷系統的瓶頸。

    ?

    ?5.回歸問題

    ?? Q:為什么崩潰前垃圾回收的時間越來越長?

    ?? A:根據內存模型和垃圾回收算法,垃圾回收分兩部分:內存標記、清除(復制),標記部分只要內存大小固定時間是不變的,變的是復制部分,因為每次垃圾回收都有一些回收不掉的內存,所以增加了復制量,導致時間延長。所以,垃圾回收的時間也可以作為判斷內存泄漏的依據

    ?? Q:為什么Full GC的次數越來越多?

    ?? A:因此內存的積累,逐漸耗盡了年老代的內存,導致新對象分配沒有更多的空間,從而導致頻繁的垃圾回收

    ?? Q:為什么年老代占用的內存越來越大?

    ?? A:因為年輕代的內存無法被回收,越來越多地被Copy到年老代

    ?

    三、性能調優

    ?除了上述內存泄漏外,我們還發現CPU長期不足3%,系統吞吐量不夠,針對8core×16G、64bit的Linux服務器來說,是嚴重的資源浪費。

    ?在CPU負載不足的同時,偶爾會有用戶反映請求的時間過長,我們意識到必須對程序及JVM進行調優。從以下幾個方面進行:

    • 線程池:解決用戶響應時間長的問題
    • 連接池
    • JVM啟動參數:調整各代的內存比例和垃圾回收算法,提高吞吐量
    • 程序算法:改進程序邏輯算法提高性能

    ? 1.Java線程池(java.util.concurrent.ThreadPoolExecutor)

    ??? 大多數JVM6上的應用采用的線程池都是JDK自帶的線程池,之所以把成熟的Java線程池進行羅嗦說明,是因為該線程池的行為與我們想象的有點出入。Java線程池有幾個重要的配置參數:

    • corePoolSize:核心線程數(最新線程數)
    • maximumPoolSize:最大線程數,超過這個數量的任務會被拒絕,用戶可以通過RejectedExecutionHandler接口自定義處理方式
    • keepAliveTime:線程保持活動的時間
    • workQueue:工作隊列,存放執行的任務

    ??? Java線程池需要傳入一個Queue參數(workQueue)用來存放執行的任務,而對Queue的不同選擇,線程池有完全不同的行為:

    • SynchronousQueue:?一個無容量的等待隊列,一個線程的insert操作必須等待另一線程的remove操作,采用這個Queue線程池將會為每個任務分配一個新線程
    • LinkedBlockingQueue?:?無界隊列,采用該Queue,線程池將忽略?maximumPoolSize參數,僅用corePoolSize的線程處理所有的任務,未處理的任務便在LinkedBlockingQueue中排隊
    • ArrayBlockingQueue:?有界隊列,在有界隊列和?maximumPoolSize的作用下,程序將很難被調優:更大的Queue和小的maximumPoolSize將導致CPU的低負載;小的Queue和大的池,Queue就沒起動應有的作用。

    ??? 其實我們的要求很簡單,希望線程池能跟連接池一樣,能設置最小線程數、最大線程數,當最小數<任務<最大數時,應該分配新的線程處理;當任務>最大數時,應該等待有空閑線程再處理該任務。

    ??? 但線程池的設計思路是,任務應該放到Queue中,當Queue放不下時再考慮用新線程處理,如果Queue滿且無法派生新線程,就拒絕該任務。設計導致“先放等執行”、“放不下再執行”、“拒絕不等待”。所以,根據不同的Queue參數,要提高吞吐量不能一味地增大maximumPoolSize。

    ??? 當然,要達到我們的目標,必須對線程池進行一定的封裝,幸運的是ThreadPoolExecutor中留了足夠的自定義接口以幫助我們達到目標。我們封裝的方式是:

    • 以SynchronousQueue作為參數,使maximumPoolSize發揮作用,以防止線程被無限制的分配,同時可以通過提高maximumPoolSize來提高系統吞吐量
    • 自定義一個RejectedExecutionHandler,當線程數超過maximumPoolSize時進行處理,處理方式為隔一段時間檢查線程池是否可以執行新Task,如果可以把拒絕的Task重新放入到線程池,檢查的時間依賴keepAliveTime的大小。

    ? 2.連接池(org.apache.commons.dbcp.BasicDataSource)

    ??? 在使用org.apache.commons.dbcp.BasicDataSource的時候,因為之前采用了默認配置,所以當訪問量大時,通過JMX觀察到很多Tomcat線程都阻塞在BasicDataSource使用的Apache ObjectPool的鎖上,直接原因當時是因為BasicDataSource連接池的最大連接數設置的太小,默認的BasicDataSource配置,僅使用8個最大連接。

    ??? 我還觀察到一個問題,當較長的時間不訪問系統,比如2天,DB上的Mysql會斷掉所以的連接,導致連接池中緩存的連接不能用。為了解決這些問題,我們充分研究了BasicDataSource,發現了一些優化的點:

    • Mysql默認支持100個鏈接,所以每個連接池的配置要根據集群中的機器數進行,如有2臺服務器,可每個設置為60
    • initialSize:參數是一直打開的連接數
    • minEvictableIdleTimeMillis:該參數設置每個連接的空閑時間,超過這個時間連接將被關閉
    • timeBetweenEvictionRunsMillis:后臺線程的運行周期,用來檢測過期連接
    • maxActive:最大能分配的連接數
    • maxIdle:最大空閑數,當連接使用完畢后發現連接數大于maxIdle,連接將被直接關閉。只有initialSize < x < maxIdle的連接將被定期檢測是否超期。這個參數主要用來在峰值訪問時提高吞吐量。
    • initialSize是如何保持的?經過研究代碼發現,BasicDataSource會關閉所有超期的連接,然后再打開initialSize數量的連接,這個特性與minEvictableIdleTimeMillis、timeBetweenEvictionRunsMillis一起保證了所有超期的initialSize連接都會被重新連接,從而避免了Mysql長時間無動作會斷掉連接的問題。

    ? 3.JVM參數

    ??? 在JVM啟動參數中,可以設置跟內存、垃圾回收相關的一些參數設置,默認情況不做任何設置JVM會工作的很好,但對一些配置很好的Server和具體的應用必須仔細調優才能獲得最佳性能。通過設置我們希望達到一些目標:

    • GC的時間足夠的小
    • GC的次數足夠的少
    • 發生Full GC的周期足夠的長

    ? 前兩個目前是相悖的,要想GC時間小必須要一個更小的堆,要保證GC次數足夠少,必須保證一個更大的堆,我們只能取其平衡。

    ?? (1)針對JVM堆的設置,一般可以通過-Xms -Xmx限定其最小、最大值,為了防止垃圾收集器在最小、最大之間收縮堆而產生額外的時間,我們通常把最大、最小設置為相同的值
    ?? (2)年輕代和年老代將根據默認的比例(1:2)分配堆內存,可以通過調整二者之間的比率NewRadio來調整二者之間的大小,也可以針對回收代,比如年輕代,通過 -XX:newSize -XX:MaxNewSize來設置其絕對大小。同樣,為了防止年輕代的堆收縮,我們通常會把-XX:newSize -XX:MaxNewSize設置為同樣大小

    ?? (3)年輕代和年老代設置多大才算合理?這個我問題毫無疑問是沒有答案的,否則也就不會有調優。我們觀察一下二者大小變化有哪些影響

    • 更大的年輕代必然導致更小的年老代,大的年輕代會延長普通GC的周期,但會增加每次GC的時間;小的年老代會導致更頻繁的Full GC
    • 更小的年輕代必然導致更大年老代,小的年輕代會導致普通GC很頻繁,但每次的GC時間會更短;大的年老代會減少Full GC的頻率
    • 如何選擇應該依賴應用程序對象生命周期的分布情況:如果應用存在大量的臨時對象,應該選擇更大的年輕代;如果存在相對較多的持久對象,年老代應該適當增大。但很多應用都沒有這樣明顯的特性,在抉擇時應該根據以下兩點:(A)本著Full GC盡量少的原則,讓年老代盡量緩存常用對象,JVM的默認比例1:2也是這個道理 (B)通過觀察應用一段時間,看其他在峰值時年老代會占多少內存,在不影響Full GC的前提下,根據實際情況加大年輕代,比如可以把比例控制在1:1。但應該給年老代至少預留1/3的增長空間

    ? (4)在配置較好的機器上(比如多核、大內存),可以為年老代選擇并行收集算法:?-XX:+UseParallelOldGC?,默認為Serial收集

    ? (5)線程堆棧的設置:每個線程默認會開啟1M的堆棧,用于存放棧幀、調用參數、局部變量等,對大多數應用而言這個默認值太了,一般256K就足用。理論上,在內存不變的情況下,減少每個線程的堆棧,可以產生更多的線程,但這實際上還受限于操作系統。

    ? (4)可以通過下面的參數打Heap Dump信息

    • -XX:HeapDumpPath
    • -XX:+PrintGCDetails
    • -XX:+PrintGCTimeStamps
    • -Xloggc:/usr/aaa/dump/heap_trace.txt

    ??? 通過下面參數可以控制OutOfMemoryError時打印堆的信息

    • -XX:+HeapDumpOnOutOfMemoryError

    ?請看一下一個時間的Java參數配置:(服務器:Linux 64Bit,8Core×16G)

    ?

    ?JAVA_OPTS="$JAVA_OPTS -server -Xms3G -Xmx3G -Xss256k -XX:PermSize=128m -XX:MaxPermSize=128m -XX:+UseParallelOldGC -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/usr/aaa/dump -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -Xloggc:/usr/aaa/dump/heap_trace.txt -XX:NewSize=1G -XX:MaxNewSize=1G"

    經過觀察該配置非常穩定,每次普通GC的時間在10ms左右,Full GC基本不發生,或隔很長很長的時間才發生一次

    通過分析dump文件可以發現,每個1小時都會發生一次Full GC,經過多方求證,只要在JVM中開啟了JMX服務,JMX將會1小時執行一次Full GC以清除引用,關于這點請參考附件文檔。

    ?4.程序算法調優:本次不作為重點

    ?

    參考資料:

    http://java.sun.com/javase/technologies/hotspot/gc/gc_tuning_6.html

    來源:http://blog.csdn.net/chen77716/article/details/5695893

    ?

    =======================================================================================

    調優方法

    一切都是為了這一步,調優,在調優之前,我們需要記住下面的原則:

    ?

    1、多數的Java應用不需要在服務器上進行GC優化;

    2、多數導致GC問題的Java應用,都不是因為我們參數設置錯誤,而是代碼問題;

    3、在應用上線之前,先考慮將機器的JVM參數設置到最優(最適合);

    4、減少創建對象的數量;

    5、減少使用全局變量和大對象;

    6、GC優化是到最后不得已才采用的手段;

    7、在實際使用中,分析GC情況優化代碼比優化GC參數要多得多;

    ?

    GC優化的目的有兩個(http://www.360doc.com/content/13/0305/10/15643_269388816.shtml):

    1、將轉移到老年代的對象數量降低到最小;

    2、減少full GC的執行時間;

    ?

    為了達到上面的目的,一般地,你需要做的事情有:

    1、減少使用全局變量和大對象;

    2、調整新生代的大小到最合適;

    3、設置老年代的大小為最合適;

    4、選擇合適的GC收集器;

    ?

    在上面的4條方法中,用了幾個“合適”,那究竟什么才算合適,一般的,請參考上面“收集器搭配”和“啟動內存分配”兩節中的建議。但這些建議不是萬能的,需要根據您的機器和應用情況進行發展和變化,實際操作中,可以將兩臺機器分別設置成不同的GC參數,并且進行對比,選用那些確實提高了性能或減少了GC時間的參數。

    ?

    真正熟練的使用GC調優,是建立在多次進行GC監控和調優的實戰經驗上的,進行監控和調優的一般步驟為:

    1,監控GC的狀態

    使用各種JVM工具,查看當前日志,分析當前JVM參數設置,并且分析當前堆內存快照和gc日志,根據實際的各區域內存劃分和GC執行時間,覺得是否進行優化;

    ?

    2,分析結果,判斷是否需要優化

    如果各項參數設置合理,系統沒有超時日志出現,GC頻率不高,GC耗時不高,那么沒有必要進行GC優化;如果GC時間超過1-3秒,或者頻繁GC,則必須優化;

    注:如果滿足下面的指標,則一般不需要進行GC:

    ? ?Minor GC執行時間不到50ms;

    ? ?Minor GC執行不頻繁,約10秒一次;

    ? ?Full GC執行時間不到1s;

    ? ?Full GC執行頻率不算頻繁,不低于10分鐘1次;

    ?

    3,調整GC類型和內存分配

    如果內存分配過大或過小,或者采用的GC收集器比較慢,則應該優先調整這些參數,并且先找1臺或幾臺機器進行beta,然后比較優化過的機器和沒有優化的機器的性能對比,并有針對性的做出最后選擇;

    4,不斷的分析和調整

    通過不斷的試驗和試錯,分析并找到最合適的參數

    5,全面應用參數

    如果找到了最合適的參數,則將這些參數應用到所有服務器,并進行后續跟蹤。

    ?

    ?

    調優實例

    上面的內容都是紙上談兵,下面我們以一些真實例子來進行說明:

    實例1:

    筆者昨日發現部分開發測試機器出現異常:java.lang.OutOfMemoryError: GC overhead limit exceeded,這個異常代表:

    GC為了釋放很小的空間卻耗費了太多的時間,其原因一般有兩個:1,堆太小,2,有死循環或大對象;

    筆者首先排除了第2個原因,因為這個應用同時是在線上運行的,如果有問題,早就掛了。所以懷疑是這臺機器中堆設置太小;

    使用ps -ef |grep "java"查看,發現:

    ?

    ?

    該應用的堆區設置只有768m,而機器內存有2g,機器上只跑這一個java應用,沒有其他需要占用內存的地方。另外,這個應用比較大,需要占用的內存也比較多;

    筆者通過上面的情況判斷,只需要改變堆中各區域的大小設置即可,于是改成下面的情況:

    ?

    ?

    跟蹤運行情況發現,相關異常沒有再出現;

    ?

    實例2:(http://www.360doc.com/content/13/0305/10/15643_269388816.shtml)

    一個服務系統,經常出現卡頓,分析原因,發現Full GC時間太長

    jstat -gcutil:

    S0 ? ? S1 ? ?E ? ? O ? ? ? P ? ? ? ?YGC YGCT FGC FGCT ?GCT

    12.16 0.00 5.18 63.78 20.32 ?54 ? 2.047 5 ? ? 6.946 ?8.993?

    分析上面的數據,發現Young GC執行了54次,耗時2.047秒,每次Young GC耗時37ms,在正常范圍,而Full GC執行了5次,耗時6.946秒,每次平均1.389s,數據顯示出來的問題是:Full GC耗時較長,分析該系統的是指發現,NewRatio=9,也就是說,新生代和老生代大小之比為1:9,這就是問題的原因:

    1,新生代太小,導致對象提前進入老年代,觸發老年代發生Full GC;

    2,老年代較大,進行Full GC時耗時較大;

    優化的方法是調整NewRatio的值,調整到4,發現Full GC沒有再發生,只有Young GC在執行。這就是把對象控制在新生代就清理掉,沒有進入老年代(這種做法對一些應用是很有用的,但并不是對所有應用都要這么做)

    ?

    實例3:

    一應用在性能測試過程中,發現內存占用率很高,Full GC頻繁,使用sudo -u admin -H ?jmap -dump:format=b,file=文件名.hprof pid 來dump內存,生成dump文件,并使用Eclipse下的mat差距進行分析,發現:


    ?

    從圖中可以看出,這個線程存在問題,隊列LinkedBlockingQueue所引用的大量對象并未釋放,導致整個線程占用內存高達378m,此時通知開發人員進行代碼優化,將相關對象釋放掉即可。

    ?

    123456789101112131415161718192021222324-Xmx??? Java Heap最大值,默認值為物理內存的1/4,最佳設值應該視物理內存大小及計算機內其他內存開銷而定;????-Xms??? Java Heap初始值,Server端JVM最好將-Xms和-Xmx設為相同值,開發測試機JVM可以保留默認值;????-Xmn??? Java Heap Young區大小,不熟悉最好保留默認值;???? -Xss??? 每個線程的Stack大小,不熟悉最好保留默認值;???22. 如何分配JVM內存設置:????(1)當在命令提示符下啟動并使用JVM時(只對當前運行的類Test生效):???? java -Xmx128m -Xms64m -Xmn32m -Xss16m Test???? (2)當在集成開發環境下(如eclipse)啟動并使用JVM時:????a. 在eclipse根目錄下打開eclipse.ini,默認內容為(這里設置的是運行當前開發工具的JVM內存分配):???? -vmargs???? -Xms40m???? -Xmx256m-vmargs表示以下為虛擬機設置參數,可修改其中的參數值,也可添加-Xmn,-Xss,另外,eclipse.ini內還可以設置非堆內存,如:-XX:PermSize=56m,-XX:MaxPermSize=128m.????此處設置的參數值可以通過以下配置在開發工具的狀態欄顯示:???? 在eclipse根目錄下創建文件options,文件內容為:org.eclipse.ui/perf/showHeapStatus=true????修改eclipse根目錄下的eclipse.ini文件,在開頭處添加如下內容:???? -debug? options???? -vm? javaw.exe????重新啟動eclipse,就可以看到下方狀態條多了JVM信息.????b. 打開eclipse-窗口-首選項-Java-已安裝的JRE(對在當前開發環境中運行的java程序皆生效)????編輯當前使用的JRE,在缺省VM參數中輸入:-Xmx128m -Xms64m -Xmn32m -Xss16m????c. 打開eclipse-運行-運行-Java應用程序(只對所設置的java類生效)???? 選定需設置內存分配的類-自變量,在VM自變量中輸入:-Xmx128m -Xms64m選定需設置內存分配的類-自變量,在VM自變量中輸入:-Xmx128m -Xms64m -Xmn32m -Xss16m????注:如果在同一開發環境中同時進行了b和c設置,則b設置生效,c設置無效,如:????開發環境的設置為:-Xmx256m,而類Test的設置為:-Xmx128m -Xms64m,則運行Test時生效的設置為:???? -Xmx256m -Xms64m????(3)當在服務器環境下(如Tomcat)啟動并使用JVM時(對當前服務器環境下所以Java程序生效):???? a. 設置環境變量:???? 變量名:CATALINA_OPTS????變量值:-Xmx128m -Xms64m -Xmn32m -Xss16m3?b. 打開Tomcat根目錄下的bin文件夾,編輯catalina.bat,將其中的%CATALINA_OPTS%(共有四處)替換為:-Xmx128m -Xms64m -Xmn32m -Xss16m

      


    超強干貨來襲 云風專訪:近40年碼齡,通宵達旦的技術人生

    總結

    以上是生活随笔為你收集整理的java虚拟机性能监控调优及原则的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。