溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java CPU性能分析工具代碼實(shí)例

發(fā)布時(shí)間:2020-09-05 05:01:18 來(lái)源:腳本之家 閱讀:174 作者:cjunn 欄目:編程語(yǔ)言

這篇文章主要介紹了Java CPU性能分析工具代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

背景

有處理過(guò)生產(chǎn)問(wèn)題的同學(xué)基本都能遇到系統(tǒng)忽然緩慢,CPU突然飆升,甚至整個(gè)應(yīng)用請(qǐng)求不可用。當(dāng)出現(xiàn)這種情況下,在不影響數(shù)據(jù)準(zhǔn)確性的前提下,我們應(yīng)該盡快導(dǎo)出jstack和內(nèi)存信息,然后重啟系統(tǒng),盡快回復(fù)系統(tǒng)的可用性,避免用戶體驗(yàn)過(guò)差。本文針對(duì)CPU飆升問(wèn)題,提供該問(wèn)題的排查思路,從而能夠快速定位到某線程甚至某快代碼導(dǎo)致CPU飆升,從而提供處理該問(wèn)題的思路。

排查過(guò)程

  • 通過(guò)top命令查看cpu飆升的java進(jìn)程pid
  • 通過(guò)ps -mp [pid] -o THREAD,tid,time查看該進(jìn)程下所擁有的線程及各個(gè)線程占用cpu的使用率,并且記錄CPU使用率過(guò)高的線程ID號(hào)
  • 將線程ID號(hào)轉(zhuǎn)換為16進(jìn)程的數(shù)值記為tid_hex
  • 使用jdk自帶jstack監(jiān)控命令
  • 使用命令jstack [pid] | grep tid_hex -A100命令輸出該線程的堆棧信息
  • 根據(jù)堆棧信息分析代碼。

通過(guò)以上步驟可以查找出導(dǎo)致cpu飆升的相關(guān)代碼位置,然后對(duì)代碼進(jìn)行code review即可。

工具封裝

以上步驟已經(jīng)封裝為腳本文件,通過(guò)以下腳本文件只需要指定進(jìn)程ID即pid即可導(dǎo)出默認(rèn)前5條導(dǎo)致CPU率過(guò)高的堆棧信息。

已上傳github : 點(diǎn)我進(jìn)入

./java-thread-top.sh -p pid
#!/bin/bash
# @Function
# Find out the highest cpu consumed threads of java processes, and print the stack of these threads.
# @github https://github.com/cjunn/script_tool/
# @author cjunn
# @date Sun Jan 12 2020 21:08:58 GMT+0800
#

pid='';
count=5;

function usage(){
  readonly PROG="`basename $0`"
  cat <<EOF
Usage: ${PROG} [OPTION]
Find out the highest cpu consumed threads of java processes,
and print the stack of these threads.
Example:
 ${PROG} -p <pid> -c 5   # show top 5 busy java threads info
Output control:
 -p, --pid <java pid>   find out the highest cpu consumed threads from
              the specified java process.
              default from all java process.
 -c, --count <num>     set the thread count to show, default is 5.
Miscellaneous:
 -h, --help        display this help and exit.
EOF
}

#1.Collect script parameters
#2.Check whether PID exists
if [ $# -gt 0 ];
then
  while true; do
    case "$1" in
    -c|--count)
      count="$2"
      shift 2
      ;;
    -p|--pid)
      pid="$2"
      shift 2
      ;;
    -h|--help)
      usage
      exit 0;
      ;;
    --)
      shift
      break
      ;;
    *)
      shift
      if [ -z "$1" ] ; then
        break
      fi
      ;;
    esac
  done
fi
if [ ! -n "$pid" ] ;then
  echo "error: -p is empty"
  exit 1;
fi

function worker(){
  #1.Query all threads according to PID.
  #2.Delete header and first line information.
  #3.According to the second column of CPU to sort, reverse display.
  #4.Delete the count + 1 to last column based on the count value.
  #5.Get CPU utilization, TID value, thread used time, and assign them to CPU, TID, time respectively.
  #6.Perform hex conversion on TID.
  #7.Use JDK to monitor all threads of jstack output PID.
  #8.Use awk to regularly query the thread information of tid_hex required.
  #9.Display the stack information of count before thread busy.
  local whilec=0;
  ps -mp $pid -o THREAD,tid,time | sed '1,2d' | sort -k 2 -n -r |sed $[$count+1]',$d' | awk '{print $2,$8,$9}' | while read cpu tid time
  do
      tid_hex=$(printf "%x" $tid);
      echo "====================== tid:${tid} tid_hex:${tid_hex} cpu:${cpu} time:${time} ======================";
      jstack $pid | awk 'BEGIN {RS = "\n\n+";ORS = "\n\n"} /'${tid_hex}'/ {print $0}'
      echo "";
      whilec=$[$whilec+1];
  done
  if [ $whilec -eq 0 ] ; then
    echo "error : thread not found, make sure pid exists.";
  fi

}
worker

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI