概述
公司內部的一個產品 (java 開發(fā)的) 運行在 window 虛擬機上鬓照,運行一段時間后CPU飆升弃甥,然后想查看是哪個線程占用惶室。
折騰了一下午斋枢,終于定位到該線程帘靡。
下面我們通過兩種方式定位到占用cpu比較高的線程。
- 使用Process Explorer瓤帚,第三方工具定位描姚,使用比較簡單,容易上手戈次。
- 使用window自帶的perfmon 性能監(jiān)控工具進行監(jiān)控轩勘,功能強大,但稍微有點復雜怯邪。
示例程序
public class ThreadCpuTest {
public static void main(String[] args) {
Thread busyTask = new Thread(() -> {
for (;;) {
double a = new Random().nextDouble();
double b = new Random().nextDouble();
double c = a * b;
}
});
busyTask.setName("busy task");
busyTask.start();
Thread lazyTask = new Thread(() -> {
for (;;) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (Exception e) {
e.printStackTrace();
}
}
});
busyTask.setName("lazy task");
lazyTask.start();
}
}
運行該示例绊寻,然后使用監(jiān)控工具定位到那個線程CPU使用率高。
1. Process Explorer 使用
- 第一步: 獲得該程序的進程ID
通過 jps 可以查看到 我們運行的程序的進程號為 “11964”
- 第二步: 打開Process Explorer工具
下面我們通過 Process Explorer 工具進行查看 該進程中所有的線程悬秉。
如果你沒有 Process Explorer 澄步,點擊下面連接,進行下載和泌。
點擊 https://docs.microsoft.com/zh-cn/sysinternals/downloads/process-explorer 地址下載 Process Explorer
打開Process Explorer 后村缸,界面如下:
- 第三步: 查找進程號為“11964”的進程
通過 jps 命令查看的進程號為 “11964”。然后在 Process Explorer中找到該進程武氓。
-
第四步: 打開該進程Properties界面
然后選中該進程王凑,雙擊或右鍵 選擇“Properties...”
-
第五步: 選中“Threads”標簽頁,查看線程統(tǒng)計信息
選中 Threads 標簽頁聋丝。
發(fā)現(xiàn)線程ID為“20024” 的線程占用cpu比較高索烹。 第六步: 線程ID轉換成十六進制
然后我們把 十進制的 “20024” 轉換為 十六進制的“4E38”。
第七步: dump線程信息
jstack 11964 > thread_dump.txt-
第八步: 查找線程ID為“4E38”的線程
從線程的名字中弱睦,我們發(fā)現(xiàn)該線程的名字是“busy_task”百姓。
然后就可以找到該線程,查看該線程是什么原因大量占用CPU况木,剩下的就是改程序的事情了
2. Perfmon 使用
-
第一步: 首先使用 jps 獲取當前程序運行的進程ID
進程ID:“8880” -
第二步: 打開perfmon 性能監(jiān)視器
-
第三步: 打開添加計數器頁面
-
第四步: 添加計數器
在左上部分找中垒拢,到Thread旬迹,選中 %Processor Time和 ID Thread 兩個選項。
在左下部分中(選定對象的實例)求类,找到所有“javaw”選中奔垦,點擊添加按鈕,然后顯示如右半部分效果尸疆。 -
第五步: 切換報告顯示方式
-
第六步: 查找使用CPU比較高的線程
從圖中發(fā)現(xiàn)線程ID為“10320”的線程椿猎,占用CPU比較高。 -
第七步: 線程ID 轉換成十六進制
線程ID 轉換成十六進制為“2850” -
第八步: dump 線程信息
-
第九步: 查找線程ID為“2850”的線程
從線程堆棧信息中可以看出該線程為“busy_task”線程寿弱。