內存redis
GC數據庫
拿到GC日誌,分析GC日誌(吞吐量,停頓時間,垃圾回收次數;這三個是評判垃圾收集器好壞的標準)後端
首先咱們須要知道系統當前的運行情況,也就是系統的性能好壞,才能判斷是否須要調優。若是系統的響應時間很短,計算機的資源使用也很低,那咱們作系統調優就徹底是爲了調優而調優。那麼衡量系統性能的指標到底有哪些呢?性能優化
具體來講 JVM 調優須要包括兩方面:合理地設置 JVM 的內存空間和選擇合適的垃圾回收器。網絡
通常項目確定是不須要進行 JVM 調優的,由於 JVM 自己就是爲這種低延時、高併發、大吞吐的服務設計和優化的,咱們不多須要去改變什麼。因此,咱們每每更偏重於應用服務自己的調優。 併發
在一些應用中,好比大數據計算引擎,是一種很是極端的 JVM 應用,對延時的要求並不高,但對吞吐量要求很高,會有大量的短生命週期對象產生,同時也有大量的對象生存時間很是久,咱們就須要對特定的一些 JVM 參數進行修改。 app
再好比生產環境中出現內存溢出,咱們須要判斷是因爲大峯值下沒有限流,瞬間建立大量對象而致使的內存溢出,仍是是因爲內存泄漏而致使的內存溢出。對於內存泄漏致使的,這種問題就是程序的 Bug,咱們須要及時找到問題代碼進行修改,而不是調整 JVM。 負載均衡
JVM 在很大程度上減輕了 Java 開發人員投入到對象生命週期管理的精力。在使用對象的時候,JVM 會自動分配內存給對象,在不使用的時候,垃圾回收器會自動回收對象,釋放佔用的內存。因此通常狀況下咱們是不須要調優的。固然事無絕對,某些特殊場景就須要咱們進行參數調整,但調整的前提必定是你對 JVM 的運行原理很是熟悉才行。異步
JVM性能優化到底從發現到解決的歷程:發現問題-排查問題-解決問題分佈式
發現問題:JVM日誌 gc.log 文件,經過JVM工具(好比:gceasy)查看並發現問題;好比GC的次數過多;能夠經過工具查看到GC次數【新生代和老年代分別的GC次數】。GC頻繁:如何判斷GC頻繁呢?有個參照【好比服務剛上線GC5次,運行一段時間後10次,在以後30次,在以後50次,依次類推】
排查問題:打印出JVM GC日誌,查看minorGC(新生代GC)或者majorGC(老年代GC)
解決問題:適當增長堆內存空間,或者選擇合適的垃圾收集器
發現問題:OOM
排查問題:在JVM參數中配置,若是發生了OOM錯誤時自動dump下相關的.hprof文件,對該文件經過工具(好比MAT或者在線的perfma)進行分析;分析以後當找到佔用內存比較大的對象對應的線程的業務代碼(多是程序死循環,或者後端程序併發量比較大)
解決問題:若是是併發量比較大,就減小對後端程序的訪問;經過Nginx增長機器,負載均衡,權重比例
發現問題:CPU負載太高
排查問題:命令:top jps jinfo jstat jmap 等這些命令靈活配合使用查看;多是服務程序處理壓力過大
解決問題:具體看狀況而論,能夠集羣部署、或者經過中間件(MQ、Kafka等)實現異步請求
發現問題:死鎖
排查問題:能夠經過 jstack 命令去查看相關線程鎖的信息
解決問題:找到對應的業務代碼,進行修改;或者使用zk、redis實現分佈式鎖
發現問題:線程池不夠用了
排查問題:經過JDK的工具 jconcole jvisualvm 查看哪些線程得不到釋放的
解決問題:適當的對後端代碼優化,及時釋放資源、合理的設置線程池中的參數(大小)