工具的进化一直是人类生产力进步的标志,合理使用工具能大大提升咱们的工做效率,遇到问题时,合理使用工具更能加快问题排查的进度。这也是我为何很是喜欢 shell 的缘由,它丰富的命令行工具集加管道特性处理起文本数据集来真的精准而优雅,让人迷醉。java
但不少时候文本的表现力很是有限,能够说匮乏,表达绝对值时,天然是无往不利,但在展现相对值时,就有些捉襟见肘了,就更不用说多维数据了。咱们用 shell 能够很是快速地查询出文本内的累加值、最大值等,但一遇到两组值的相关性分析时,就一筹莫展了。这个时候,就须要使用另外一种分析工具 – 图
了,如散点图就能很清晰地展现相关性。git
今天就准备介绍一种图,火焰图
,以前组内大神分享过它的使用办法,但我以后好久都没有用过,以致于对它没有什么深入印象,最近排查咱们 Java 应用负载问题时试用了一下,这才对它的用途有了点心得。github
转载随意,文章会持续修订,请注明来源地址:https://zhenbianshu.github.io 。shell
在排查性能问题时,咱们一般会把线程栈 dump 出来,而后使用 grep --no-group-separator -A 1 java.lang.Thread.State jstack.log | awk 'NR%2==0' | sort | uniq -c | sort -nr
相似的 shell 语句,查看大多数线程栈都在干什么。而由线程栈的出现频率,来推断 JVM 内耗时最多的调用。apache
至于其原理,设想广场上有一个大屏幕在不停地播放各类广告。若是咱们随机对大屏幕拍照,次数多了,统计照片中各个广告出现的频率,基本能够得出每一个广告的播放时长占比了。而咱们应用的资源就像大屏幕,每次调用就像是播放一次广告,统计 dump 出的线程栈出现比例,也就基本能看出线程栈的耗时占比,虽然有偏差,可是屡次统计下应该差不了多少。这也就是为何有些家长每次进孩子房间都发现孩子在看系统桌面后觉得孩子平时喜欢对着桌面发呆的缘由。 :)bash
2444 at org.apache.catalina.loader.WebappClassLoaderBase.loadClass(WebappClassLoaderBase.java:1200) 1587 at sun.misc.Unsafe.park(Native Method) 795 at java.security.Provider.getService(Provider.java:1035) 293 at java.lang.Object.wait(Native Method) 292 at java.lang.Thread.sleep(Native Method) 73 at org.apache.logging.log4j.core.layout.TextEncoderHelper.copyDataToDestination(TextEncoderHelper.java:61) 71 at sun.nio.ch.EPollArrayWrapper.epollWait(Native Method) 70 at java.lang.Class.forName0(Native Method) 54 at org.apache.logging.log4j.core.appender.rolling.RollingFileManager.checkRollover(RollingFileManager.java:217)
可是这样有些问题,首先写 shell 挺费事的,另外若是我想查看自栈顶第二个栈的最多调用,即便修改了 shell 命令,结果也不直观。网络
产生这个问题的主要缘由是,咱们的线程栈是有调用关系的,即咱们须要考虑线程栈的 调用链
和 出现频率
两个维度,而单一的文本表现这两种维度比较困难,因此,著名性能分析大师 brendan gregg 就提出了火焰图。app
火焰图,因其形似火焰而得名,其开源代码地址在 Github-brendangregg-Flamegraph。ide
它是一种 svg 可交互式图形,咱们经过点击和鼠标指向能够展现出更多的信息。下图就是一个典型的火焰图,从结构上,它是由多个大小和颜色各异的方块构成,每一个方块上都有字符,它们底部链接在一块,组成火焰的基底,顶部分出许多”小火苗”。svg
当咱们点击方块时,图片会从咱们点击的方块为基底向上展开,而咱们鼠标指向方块时,会展现出方块的详细说明。
介绍火焰图的分析前,咱们要首先说明它的特性:
那么,给咱们一张火焰图,咱们怎么能看出系统哪里有问题呢?
由上文中的火焰图特性特性,查看火焰图时,咱们最主要的关注点要放在方块的宽度上,由于宽度表明了调用栈在全局出现的次数,次数表明着出现频率,而频率也就能够说明耗时。
可是观察火焰图底部或中部方块的宽度占比意义不大,如上面的火焰图,中部的 do_redirections
函数宽度是 24.87%,也就是说它耗用了整个应用近四分之一的时间,可是真正消耗时间的并非 do_redirections 函数,而是 do_redirections 内部又调用的其余函数,而它的子调用分为了不少个,每一个调用的耗时并无异常。
咱们更应该关注的是火焰图顶部的一些 “平顶山”,顶部说明它没有子调用,方块宽说明它耗时长,长时间 hang 住,或者被很是频率地调用,这种方块指向的调用才是性能问题的罪魁祸首。
找到了异常调用,直接优化它,或者再根据火焰图的调用链层层向下,找到咱们的业务代码进行优化,也就大功告成。
每种工具都有其适合的应用场景,火焰图则适合用在:
既然火焰图这么强大,那么咱们该怎么实现呢?
brendan gregg 大神已经把生成火焰图的方法用 perl 实现了,开源代码就在上文的 Github 仓库中,根目录下的 flamegraph.pl
文件就是可执行的 perl 文件了。
这个命令还能够传入各类参数,支持咱们修改火焰图的颜色、大小等 。
但 flamegraph.pl 只能处理特定格式的文件,像:
a;b;c 12
a;d 3
b;c 3
z;d 5
a;c;e 3
前面是调用链,每一个调用之间用 ;
隔开,每行后面的数字是调用栈出现的次数。
如上面的数据,用 flamegraph.pl 生成的火焰图以下图:
至于咱们的 jstack 信息如何被处理成上面的格式,大神则为常见的 dump 格式都提供了工具,像 stackcollapse-perf.pl
能够处理 perf
命令的输出,stackcollapse-jstack.pl
处理 jstack
输出,stackcollapse-gdb.pl
处理 gdb 输出的栈等。
也能够用 shell 简单地实现一下 jstack 的处理方式:
grep -v -P '.+prio=\d+ os_prio=\d+' | grep -v -E 'locked <' | awk '{if ($0==""){print $0}else{printf"%s;",$0}}' | sort | uniq -c | awk '{a=$1;$1="";print $0,a}'
火焰图总结完了,之后再遇到性能问题又多了一种应对方式。
作开发越久,越能感觉获得工具的重要性,因此我准备加一个专题来专门介绍我使用的各类工具。固然,这也就更须要我更多地了解、使用和总结新的工具了。
关于本文有什么疑问能够在下面留言交流,若是您以为本文对您有帮助,欢迎关注个人 微博 或 GitHub 。您也能够在个人 博客REPO 右上角点击 Watch
并选择 Releases only
项来 订阅
个人博客,有新文章发布会第一时间通知您。