Java進程CPU使用率過高排查步驟詳細講解
前言
當(dāng)Java應(yīng)用進程出現(xiàn)CPU使用率長期處于100%或異常波動時,需要系統(tǒng)性地排查原因。本文將從初步定位、中級分析到代碼調(diào)優(yōu),分步驟展示排查流程。
一、初步定位問題
1.1 確認進程狀態(tài)
# 查看進程CPU占用前十 ps -eo %cpu,pid,args | sort -nr | head -n 10 # 實時監(jiān)控進程CPU和內(nèi)存(htop需提前安裝) htop
1.2 確定Java進程ID
# 列出所有Java進程 jps -qvp # 記錄目標進程PID(示例PID=12345)
1.3 快速生成線程堆棧
jstack 12345 > thread_dump_1.txt # 重復(fù)采集多份堆棧(間隔1~5秒) sleep 3 && jstack 12345 > thread_dump_2.txt
二、分析線程狀態(tài)
2.1 識別異常線程
# 查找處于RUNNABLE狀態(tài)的線程數(shù) grep "_tid_" thread_dump_1.txt | grep RUNNABLE | wc -l # 搜索特定關(guān)鍵字(如阻塞、死鎖) grep -i -A 20 'blocked|waiting|infinite loop' thread_dump_1.txt
2.2 對比多份堆棧
# 定位持續(xù)活躍的線程(示例線程ID) comm thread_dump_1.txt thread_dump_2.txt | grep '==> 0x'
2.3 熱點方法識別(常見模式)
行為表現(xiàn) | 可能原因 | 代碼特征 |
---|---|---|
單線程持續(xù)RUNNABLE | 死循環(huán)/遞歸調(diào)用/無限循環(huán) | while(true); sleep(0) |
線程創(chuàng)建激增 | 過度使用線程池/無限線程創(chuàng)建 | new Thread().start() |
GC頻繁觸發(fā) | 內(nèi)存泄漏/對象生成過快 | 對象池未釋放/臨時對象堆積 |
同步阻塞 | 線程間互斥競爭 | synchronized塊/ReentrantLock |
定時任務(wù)激增 | 定時調(diào)度間隔設(shè)置不合理 | ScheduledExecutor的間隔過小 |
三、深度分析維度
3.1 判斷GC影響
# 檢測GC頻率(示例每秒執(zhí)行超過5次需警惕) jstat -gcutil 12345 1000 10 # 分析GC日志(需提前開啟-verbose:gc) grep GC `jinfo -flag LogFile 12345` # 定位日志路徑
分析堆外內(nèi)存(Direct Buffer)
jmap -heap 12345 | grep 'direct memory' # 若DirectCount持續(xù)增長需檢查: # NIO類庫使用情況 # 緩沖區(qū)未釋放代碼片段
3.3 代碼級逐層穿透分析
# 使用火焰圖定位熱點方法 jstack 12345 | stackcollapse.pl | flamegraph.pl > flamegraph.svg # 或使用VisualVM/JMC進行實時分析 # 右鍵目標進程 → Profiling → CPU → 開始采樣
四、代碼級診斷要點
4.1 死循環(huán)檢測方法
// 高危模式 while(yourFlag){ // 可能未修改yourFlag的邏輯 } // 正確實踐 AtomicBoolean flag = new AtomicBoolean(true); while(flag.get()){ // 允許外部修改flag Thread.yield(); // 主動釋放CPU }
4.2 線程池問題排查
// 危險配置示例 Executors.newCachedThreadPool(); // 激增線程池 // 優(yōu)化配置 ThreadPoolExecutor( 10, 100, 60L, TimeUnit.SECONDS, new SynchronousQueue<>(), new ThreadPoolExecutor.CallerRunsPolicy() )
4.3 定時任務(wù)優(yōu)化
// 不合理用法 scheduleWithFixedDelay(task, 0, 1, MILLISECONDS); // 毫秒級并發(fā) // 合理調(diào)整 scheduleWithFixedDelay(task, 500, 500, MILLISECONDS) // 節(jié)流控制
五、預(yù)防與監(jiān)控建議
5.1 運維優(yōu)化策略
- 設(shè)置線程數(shù)/隊列/內(nèi)存閾值告警
- 預(yù)設(shè)JVM參數(shù)優(yōu)化(如:
-XX:+UseG1GC -XX:+AggressiveOpts -XX:+UseBiasedLocking
- 定期執(zhí)行 jinfo -flags 驗證JVM配置
5.2 代碼規(guī)范要求
- 禁止使用Thread.sleep(0)調(diào)度邏輯
- 定時任務(wù)間隔不低于100ms
- 大數(shù)據(jù)量處理需分批次處理
- NIO緩沖區(qū)及時release()
通過以上系統(tǒng)化排查流程,可以定位大部分CPU過高的問題源。復(fù)雜場景應(yīng)結(jié)合多種工具交叉驗證,并建立完善的監(jiān)控體系實現(xiàn)預(yù)防性優(yōu)化。
總結(jié)
到此這篇關(guān)于Java進程CPU使用率過高排查步驟詳細講解的文章就介紹到這了,更多相關(guān)Java進程CPU使用率過高內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Jmeter自定義函數(shù)base64加密實現(xiàn)過程解析
這篇文章主要介紹了Jmeter自定義函數(shù)base64加密實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-07-07Java Fluent Mybatis 聚合查詢與apply方法詳解流程篇
Java中常用的ORM框架主要是mybatis, hibernate, JPA等框架。國內(nèi)又以Mybatis用的多,基于mybatis上的增強框架,又有mybatis plus和TK mybatis等。今天我們介紹一個新的mybatis增強框架 fluent mybatis關(guān)于聚合查詢、apply方法詳解2021-10-10Spring Cloud動態(tài)配置刷新RefreshScope使用示例詳解
這篇文章主要為大家介紹了Spring Cloud動態(tài)配置刷新RefreshScope使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08詳解MyEclipse中搭建spring-boot+mybatis+freemarker框架
這篇文章主要介紹了詳解MyEclipse中搭建spring-boot+mybatis+freemarker框架,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-10-10SpringBoot yaml中的數(shù)組類型取值方式
這篇文章主要介紹了SpringBoot yaml中的數(shù)組類型取值方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09