由于在项目中遇到一种情况,某段代码在进行单元测试和在 tomcat 容器中运行的性能相差数百倍,因此需要分析在不同环境下某个方法执行的具体时间,从而确定问题。Arthas 可以做到无侵入的监控应用远行情况。
安装:
wget https://alibaba.github.io/arthas/arthas-boot.jar
java -jar arthas-boot.jarlinux 下直接执行,window 下载文件后执行。

执行完成后,显示当前 path 中指定的 JDK 中正在运行的 java 进程输入相应序号,进入 sh 命令,表明已连接成功。

Arthas 命令初步使用,大致分为 5 类:
基础命令
jvm相关
class/classloader相关
monitor/watch/trace 相关
请注意,这些命令,都通过字节码增强技术来实现的,会在指定类的方法中插入一些切面来实现数据统计和观测,因此在线上、预发使用时,请尽量明确需要观测的类、方法以及条件,诊断结束要执行 shutdown 或将增强过的类执行 reset 命令。
options

通过图示表明调用 MongoTemplate.executeFindMultiInternal() 方法时,最耗时的方法是在 doWith() 方法,总共执行 10000 次,耗时 ==252.3064ms==,最少一次调用耗时 ==0.0132ms==,最大一次耗时 ==38.4329ms==,分析缘由还是在于数据量太大,MongoTemplate 通过循环遍历出结果在进行序列化。
jad com.sankuai.inf.leaf.common.ZeroIDGen
watch com.sankuai.inf.leaf.server.service.SegmentService getId '{params, target, returnObj}' -x 2params 表明入参,target 表明当前的类,returnObj 表明返回值。

stack com.sankuai.inf.leaf.server.service.SegmentService getId
sc -d com.sankuai.inf.leaf.server.service.SegmentService getId
thread b

可以看出当前线程正在等待 ReentrantLock$NonfairSync@118f1fb4,而持有这个对象的线程又在等待当前线程释放,从而形成死锁!

先放一张官方的总结大图:

总结:Arthas 是一个很优秀的 java 诊断工具,无论是安装还是使用都很简洁,并且使用文档全面、清晰明了,值得好好研究一番。
作者 | agmtopy
本文为阿里云原创内容,未经允许不得转载。