JVM故障分析及性能优化系列之一:jstack定位线程堆栈信息

前一段时间上线的系统升级之后,出现了严重的高CPU的问题,于是开始了一系列的优化处理之中,现在将这个过程做成一个系列的文章。

基本概念Basic-info

在对Java内存泄漏进行分析的时候,需要对jvm运行期间的内存占用、线程执行等情况进行记录的dump文件,常用的主要有thread dump和heap dump。

  • thread dump 主要记录JVM在某一时刻各个线程执行的情况,以栈的形式显示,是一个文本文件。通过对thread dump文件可以分析出程序的问题出现在什么地方,从而定位具体的代码然后进行修正。thread dump需要结合占用系统资源的线程id进行分析才有意义。
  • heap dump 主要记录了在某一时刻JVM堆中对象使用的情况,即某个时刻JVM堆的快照,是一个二进制文件,主要用于分析哪些对象占用了太对的堆空间,从而发现导致内存泄漏的对象。
    上面两种dump文件都具有实时性,因此需要在服务器出现问题的时候生成,并且多生成几个文件,方便进行对比分析。下面我们先来说一下如何生成 thread dump。

使用jstack生成thread dump

当服务器出现高CPU的时候,首先执行 top -c 命令动态显示进程及占用资源的排行,如下图:
top-c
top后面的参数-c可以显示进程详细的信息。top命令执行的时候还可以执行一些快捷键:

  • 1 对于多核服务器,可以显示各个CPU占用资源的情况
  • shift+h 显示所有的线程信息
  • shift+w 将当前 top 命令的设置保存到 ~/.toprc 文件中,这样不用每次都执行快捷键了

以上图为例,pid为1503的进程占用了大量的CPU资源,接下来需要将占用CPU最高进程中的线程打印出来,可以用 top -bn1 -H -p 命令,执行结果如下:
top -bn1 -H -p
上面 -bn1 参数的含义是只输出一次结果,而不是显示一个动态的结果。

我个人请喜欢用 ps -mp <pid> -o THREAD,tid,time | sort -k2r命令查看,后面的sort参数根据线程占用的cpu比例进行排序,结果如下:
ps -mp  -o THREAD,tid,time | sort -k2r
接下来我们清楚今天的主角 jstack,这是一个在JDK5开始提供的内置工具,可以打印指定进程中线程运行的状态,包括线程数量、是否存在死锁、资源竞争情况和线程的状态等等。有下面的几个常用的参数:

  • -l长列表,打印关于锁的附加信息
  • -m打印java和jni框架的所有栈信息
    因为thread id在栈信息中是以十六进制的形式显示的,因此需要使用 printf "%x \n" <tid>命令将现场id转成十六进制的值,然后执行 jstack -l <pid> | grep <thread-hex-id> -A 10 命令显示出错的堆栈信息,如下图:
    -a10

上面命令中 -A 10 参数用来指定显示行数,否则只会显示一行信息。

这样通过上图,可以很快地定位到程序问题的代码,然后对代码进行分析和改进即可。注意:需要在多个时间段提出多个 Thread Dump信息,然后综合进行对比分析,单独分析一个文件是没有意义的。

生成shell文件

上面讲述了整个的分析过程,不过所有的命令就是实时的,所以最好创建一个shell脚本瞬间执行完成,下面对 当CPU飙高时,它在做什么 这篇文章中所提供的shell进行了改进如下:

#!/bin/bash
if [ $# -le 0 ]; then
    echo "usage: $0 <pid> [line-number]"
    exit 1
fi

# java home
if test -z $JAVA_HOME 
then
    JAVA_HOME='/usr/local/jdk'
fi

#pid
pid=$1
# checking pid
if test -z "$($JAVA_HOME/bin/jps -l | cut -d '' -f 1 | grep $pid)"
then
    echo "process of $pid is not exists"
    exit
fi

#line number
linenum=$2
if test -z $linenum
then
    linenum=10
fi

stackfile=stack$pid.dump
threadsfile=threads$pid.dump

# generate java stack
$JAVA_HOME/bin/jstack -l $pid >> $stackfile
ps -mp $pid -o THREAD,tid,time | sort -k2r | awk '{if ($1 !="USER" && $2 != "0.0" && $8 !="-") print $8;}' | xargs printf "%x\n" >> $threadsfile
tids="$(cat $threadsfile)"
for tid in $tids
do
    echo "------------------------------ ThreadId ($tid) ------------------------------"
    cat $stackfile | grep 0x$tid -A $linenum
done

rm -f $stackfile $threadsfile

下一篇文章将要讲述如何对jstack生成的文件进行分析

  • 19
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: JVM故障诊断与性能优化是一本关于Java虚拟机(JVM故障排除和性能优化的高清PDF。它涵盖了JVM的基本原理以及在实际应用中可能遇到的故障和性能问题的解决方法。 首先,本书介绍了JVM的结构和工作原理,包括内存管理、垃圾收集、线程管理等方面。读者可以了解JVM是如何执行Java程序并管理资源的。 其次,本书重点讲解了JVM故障排查的方法和工具。它介绍了常见的JVM故障类型,如内存溢出、线程死锁等,并提供了相应的解决方法。此外,本书还介绍了一些常用的JVM排查工具,如jstack、jmap等,以帮助读者更好地分析定位故障。 最后,本书还介绍了JVM性能优化的基本原则和方法。它讲解了如何分析应用的性能瓶颈,并提出了相应的优化建议。读者可以学习到如何合理配置JVM参数、如何优化代码和算法等技巧,以提升应用的性能和响应速度。 总之,JVM故障诊断与性能优化是一本帮助读者解决Java应用中JVM故障和性能问题的实用指南。无论是初学者还是有一定经验的开发者,都能从中学到宝贵的知识和技巧,进一步提升自己在JVM故障排查和性能优化方面的能力。 ### 回答2: 《高清JVM故障诊断与性能优化》是一本介绍JVM故障诊断和性能优化的书籍。JVMJava虚拟机)是Java语言的核心组成部分,负责将Java程序转化为可以在不同平台上运行的字节码程序。本书主要针对JVM在实际运行过程中可能出现的故障进行了深入探讨,并提供了一些性能优化的方法和技巧。 首先,本书介绍了JVM的基本架构和原理,包括类加载器、运行时数据区和垃圾收集器等。读者可以通过了解这些基本概念,更好地理解JVM的工作方式,从而更好地进行故障诊断和性能优化。 其次,本书介绍了一些常见的JVM故障现象和原因,比如内存溢出、死锁和线程问题等。对于每一种故障,书中都给出了具体的排查方法和解决方案,帮助读者快速解决实际中遇到的问题。 最后,本书重点介绍了JVM性能优化的方法和技巧。性能优化是一个复杂而庞大的工程,需要从多个方面入手,包括内存、线程和锁等。本书通过具体的案例分析和实践经验,向读者介绍了一些有效的性能优化方法,帮助读者提升程序的性能和响应速度。 总之,《高清JVM故障诊断与性能优化》是一本全面介绍JVM故障诊断和性能优化的书籍,对于深入了解JVM工作原理,解决JVM故障和优化JVM性能具有很大的帮助。无论是对于Java开发人员还是系统管理员来说,本书都是一本值得阅读的参考书目。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值