L
L
LearnJava
Search…
Java诊断工具Arthas

1. Arthas简介

Arthas是阿里开源的一个线上java诊断工具,发现阿里还是挺喜欢开源一些技术的,造福人类。昨天试用了一下,发现真是强大,解决了我工作两年的很多困扰,有点相见恨晚的感觉。
根据官网的说明,它能解决下面的问题:
当你遇到以下类似问题而束手无策时,Arthas可以帮助你解决:
    1.
    这个类从哪个 jar 包加载的?为什么会报各种类相关的 Exception?
    2.
    我改的代码为什么没有执行到?难道是我没 commit?分支搞错了?
    3.
    遇到问题无法在线上 debug,难道只能通过加日志再重新发布吗?
    4.
    线上遇到某个用户的数据处理有问题,但线上同样无法 debug,线下无法重现!
    5.
    是否有一个全局视角来查看系统的运行状况?
    6.
    有什么办法可以监控到JVM的实时运行状态?
Arthas采用命令行交互模式,同时提供丰富的 Tab 自动补全功能,进一步方便进行问题的定位和诊断。

2. Arthas安装

在Linux下直接执行curl -L https://alibaba.github.io/arthas/install.sh | sh,然后执行./as.sh就可以进入到Arthas的命令行界面。

3. Arthas试用

安装完后试用了它的监控相关的功能,以后调试就不那么麻烦了。
1
# 进入命令行界面,刚进入的时候会显示所有的java进程,选择对应的数字然后回车就可以进入到对应的进程中
2
./as.sh
3
Arthas script version: 3.0.4.1
4
Found existing java process, please choose one and hit RETURN.
5
[1]: 27073 org.apache.rocketmq.namesrv.NamesrvStartup
6
* [2]: 2379 org.apache.catalina.startup.Bootstrap
7
2
8
Calculating attach execution time...
9
Attaching to 2379 using version 3.0.5...
10
11
real 0m0.249s
12
user 0m0.208s
13
sys 0m0.025s
14
Attach success.
15
telnet connecting to arthas server... current timestamp is 1545738424
16
Trying 127.0.0.1...
17
Connected to 127.0.0.1.
18
Escape character is '^]'.
19
,---. ,------. ,--------.,--. ,--. ,---. ,---.
20
/ O \ | .--. ''--. .--'| '--' | / O \ ' .-'
21
| .-. || '--'.' | | | .--. || .-. |`. `-.
22
| | | || |\ \ | | | | | || | | |.-' |
23
`--' `--'`--' '--' `--' `--' `--'`--' `--'`-----'
24
25
26
wiki: https://alibaba.github.io/arthas
27
version: 3.0.4
28
pid: 2379
29
timestamp: 1545738424462
30
31
$
Copied!

3.1 monitor命令

monitor可以对某个类的某个方法进行周期性监控,来输出方法的平均响应时间,成功失败次数,调用次数等。
1
# 每3秒统计queryCatalog方法的调用次数,成功失败次数和平均响应时间等。
2
$ monitor -c 3 com.timeline.services.CatalogService queryCatalog
3
Press Ctrl+C to abort.
4
Affect(class-cnt:1 , method-cnt:1) cost in 82 ms.
5
6
7
timestamp class method total success fail avg-rt(ms) fail-rate
8
----------------------------------------------------------------------------------------------------------------------
9
2018-12-25 19:57:41 com.timeline.services.CatalogService queryCatalog 3 3 0 2.46 0.00%
10
11
timestamp class method total success fail avg-rt(ms) fail-rate
12
----------------------------------------------------------------------------------------------------------------------
13
2018-12-25 19:57:44 com.timeline.services.CatalogService queryCatalog 0 0 0 0.00 0.00%
14
15
timestamp class method total success fail avg-rt(ms) fail-rate
16
----------------------------------------------------------------------------------------------------------------------
17
2018-12-25 19:57:47 com.timeline.services.CatalogService queryCatalog 0 0 0 0.00 0.00%
18
19
timestamp class method total success fail avg-rt(ms) fail-rate
20
----------------------------------------------------------------------------------------------------------------------
21
2018-12-25 19:57:50 com.timeline.services.CatalogService queryCatalog 2 2 0 2.08 0.00%
Copied!

3.2 watch命令

watch可以监控某个方法的出入参,异常信息等。
1
# 观察方法的出入参
2
# -x表示遍历深度,可以调整来打印具体的参数和结果内容,默认值是1。
3
$watch com.timeline.services.CatalogService queryCatalog "{params,returnObj}" -x 2
4
Press Ctrl+C to abort.
5
Affect(class-cnt:1 , method-cnt:1) cost in 73 ms.
6
ts=2018-12-25 19:49:55;[email protected][
7
@Object[][isEmpty=true;size=0],
8
@ArrayList[
9
@CatalogVO[CatalogVO(id=59, name=wangjun, location=beijing, time=1537256454000, description=desccc)],
10
],
11
]
12
# 可以看到入参是空的,出参是一个list
Copied!

3.3 trace命令

trace可以输出方法内部调用关系,并打印方法内部每个调用的耗时。
1
# 打印queryCatalog方法内每个调用的耗时
2
$ trace com.timeline.services.CatalogService queryCatalog
3
Press Ctrl+C to abort.
4
Affect(class-cnt:1 , method-cnt:1) cost in 101 ms.
5
`---ts=2018-12-25 20:00:25;thread_name=http-nio-8080-exec-8;id=1b;is_daemon=true;priority=5;[email protected]a3
6
`---[2.896979ms] com.timeline.services.CatalogService:queryCatalog()
7
+---[2.605774ms] com.timeline.dao.CatalogDao:queryCatalog()
8
+---[0.0048ms] java.util.ArrayList:<init>()
9
+---[min=0.001154ms,max=0.00634ms,total=0.007494ms,count=2] java.util.List:size()
10
+---[0.004428ms] java.util.List:get()
11
+---[0.005115ms] com.timeline.entity.Catalog:getId()
12
+---[0.003107ms] com.timeline.entity.Catalog:getName()
13
+---[0.0026ms] com.timeline.entity.Catalog:getLocation()
14
+---[0.002573ms] com.timeline.entity.Catalog:getTime()
15
+---[0.005901ms] com.timeline.common.Utils:timestamp2Long()
16
+---[0.002543ms] com.timeline.entity.Catalog:getDescription()
17
+---[0.005133ms] com.timeline.vo.CatalogVO:<init>()
18
`---[0.004174ms] java.util.List:add()
Copied!

3.4 stack命令

stack可以输出方法的整个调用路径。
1
# 打印queryCatalog方法的整个调用路径(不抛出堆栈也能看到调用链啦!)
2
$ stack com.timeline.services.CatalogService queryCatalog
3
Press Ctrl+C to abort.
4
Affect(class-cnt:1 , method-cnt:1) cost in 82 ms.
5
ts=2018-12-25 20:02:56;thread_name=http-nio-8080-exec-9;id=1c;is_daemon=true;priority=5;[email protected]a3
6
@com.timeline.controllers.CalalogController.queryCatalog()
7
at sun.reflect.GeneratedMethodAccessor46.invoke(null:-1)
8
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
9
at java.lang.reflect.Method.invoke(Method.java:498)
10
at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:205)
11
at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:133)
12
at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:97)
13
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:827)
14
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:738)
15
at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:85)
16
at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:967)
17
at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:901)
Copied!

3.5 tt命令

记录方法每次的调用出入参和异常信息,并且完整保留调用现场,可以再次模拟调用。
1
# 记录每次调用queryCatalog的情况,可以根据其他命令打印出入餐和复现现场
2
$ tt -t com.timeline.services.CatalogService queryCatalog
3
Press Ctrl+C to abort.
4
Affect(class-cnt:1 , method-cnt:1) cost in 128 ms.
5
6
7
INDEX TIMESTAMP COST(ms) IS-RET IS-EXP OBJECT CLASS METHOD
8
-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
9
1008 2018-12-25 20:04:46 4.747987 true false 0x2577a8a4 CatalogService queryCatalog
10
1009 2018-12-25 20:04:47 2.270835 true false 0x2577a8a4 CatalogService queryCatalog
11
1010 2018-12-25 20:04:47 1.434482 true false 0x2577a8a4 CatalogService queryCatalog
12
1011 2018-12-25 20:04:48 1.82891 true false 0x2577a8a4 CatalogService queryCatalog
13
1012 2018-12-25 20:04:49 2.27361 true false 0x2577a8a4 CatalogService queryCatalog
Copied!
当然除此之外还有很丰富的其他功能,比如JVM,ClassLoader相关的命令。
Last modified 1yr ago