目录
背景
参数说明
参考
背景
watch能方便的观察到指定函数的调用情况。能观察到的范围为:返回值
、抛出异常
、入参
,通过编写 OGNL 表达式进行对应变量的查看。ognl学习,可以参考:https://xiaopanjia.blog.csdn.net/article/details/130429470
参数说明
watch 的参数比较多,主要是因为它能在 4 个不同的场景观察对象
参数名称 | 参数说明 |
---|---|
class-pattern | 类名表达式匹配 |
method-pattern | 函数名表达式匹配 |
express | 观察表达式,默认值:{params, target, returnObj} |
condition-express | 条件表达式 |
[b] | 在函数调用之前观察 |
[e] | 在函数异常之后观察 |
[s] | 在函数返回之后观察 |
[f] | 在函数结束之后(正常返回和异常返回)观察 |
[E] | 开启正则表达式匹配,默认为通配符匹配 |
[x:] | 指定输出结果的属性遍历深度,默认为 1,最大值是 4 |
[m <arg>] |
指定 Class 最大匹配数量,默认值为 50。长格式为[maxMatch <arg>] 。 |
[n] | 设置执行的次数 |
特别说明:
1、注意函数入参
和函数出参
的区别,有可能在中间被修改导致前后不一致,除了 -b
事件点 params
代表函数入参外,其余事件都代表函数出参
2、当使用 -b
时,由于观察事件点是在函数调用前,此时返回值或异常均不存在
3、4 个观察事件点 -b
、-e
、-s
默认关闭,-f
默认打开,当指定观察点被打开后,在相应事件点会对观察表达式进行求值并输出
例如:
1、观察函数调用返回时的参数、this 对象和返回值
$ watch demo.MathGame primeFactors -x 2
method=demo.MathGame.primeFactors location=AtExit
ts=2021-08-31 15:22:58; [cost=1.020982ms] result=@ArrayList[
@Object[][
@Integer[1],
],
@MathGame[
random=@Random[java.util.Random@31cefde0],
illegalArgumentCount=@Integer[44],
],
@ArrayList[
@Integer[2],
@Integer[2],
@Integer[26947],
],
]
2、观察函数调用入口的参数和返回值
$ watch demo.MathGame primeFactors "{params,returnObj}" -x 2 -b
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 50 ms.
ts=2018-12-03 19:23:23; [cost=0.0353ms] result=@ArrayList[
@Object[][
@Integer[-1077465243],
],
null,
]
事件点为函数执行前,因此获取不到返回值。
3、同时观察函数调用前和函数返回后
$ watch demo.MathGame primeFactors "{params,target,returnObj}" -x 2 -b -s -n 2
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 46 ms.
ts=2018-12-03 19:29:54; [cost=0.01696ms] result=@ArrayList[
@Object[][
@Integer[1],
],
@MathGame[
random=@Random[java.util.Random@522b408a],
illegalArgumentCount=@Integer[13038],
],
null,
]
ts=2018-12-03 19:29:54; [cost=4.277392ms] result=@ArrayList[
@Object[][
@Integer[1],
],
@MathGame[
random=@Random[java.util.Random@522b408a],
illegalArgumentCount=@Integer[13038],
],
@ArrayList[
@Integer[2],
@Integer[2],
@Integer[2],
@Integer[5],
@Integer[5],
@Integer[73],
@Integer[241],
@Integer[439],
],
]
特殊说明:
-
参数里
-n 2
,表示只执行两次 -
这里输出结果中,第一次输出的是函数调用前的观察表达式的结果,第二次输出的是函数返回后的表达式的结果
-
结果的输出顺序和事件发生的先后顺序一致,和命令中
-s -b
的顺序无关
4、条件表达式的例子
只有满足条件的调用,才会有响应
$ watch demo.MathGame primeFactors "{params[0],target}" "params[0]<0"
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 68 ms.
ts=2018-12-03 19:36:04; [cost=0.530255ms] result=@ArrayList[
@Integer[-18178089],
@MathGame[demo.MathGame@41cf53f9],
]
5、观察异常信息的例子
$ watch demo.MathGame primeFactors "{params[0],throwExp}" -e -x 2
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 62 ms.
ts=2018-12-03 19:38:00; [cost=1.414993ms] result=@ArrayList[
@Integer[-1120397038],
java.lang.IllegalArgumentException: number is: -1120397038, need >= 2
at demo.MathGame.primeFactors(MathGame.java:46)
at demo.MathGame.run(MathGame.java:24)
at demo.MathGame.main(MathGame.java:16)
,
]
特别说明:
-
-e
表示抛出异常时才触发 - express 中,表示异常信息的变量是
throwExp
6、按照耗时进行过滤
$ watch demo.MathGame primeFactors '{params, returnObj}' '#cost>200' -x 2
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 66 ms.
ts=2018-12-03 19:40:28; [cost=2112.168897ms] result=@ArrayList[
@Object[][
@Integer[1],
],
@ArrayList[
@Integer[5],
@Integer[428379493],
],
]
-
#cost>200
(单位是ms
)表示只有当耗时大于 200ms 时才会输出,过滤掉执行时间小于 200ms 的调用
7、观察当前对象中的属性
查看函数运行前后,当前对象中的属性,可以使用target
关键字,代表当前对象。然后使用target.field_name
访问当前对象的某个属性。
$ watch demo.MathGame primeFactors 'target'
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 52 ms.
ts=2018-12-03 19:41:52; [cost=0.477882ms] result=@MathGame[
random=@Random[java.util.Random@522b408a],
illegalArgumentCount=@Integer[13355],
]
$ watch demo.MathGame primeFactors 'target.illegalArgumentCount'
Press Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 67 ms.
ts=2018-12-03 20:04:34; [cost=131.303498ms] result=@Integer[8]
ts=2018-12-03 20:04:35; [cost=0.961441ms] result=@Integer[8]
8、排除掉指定的类
使用 --exclude-class-pattern
参数可以排除掉指定的类,比如:
watch javax.servlet.Filter * --exclude-class-pattern com.demo.TestFilter
9、投影(Across)
简单的说,就是查看集合中元素的属性或者方法返回值
watch com.example.httpclientdemo.HttpclientDemoApplication doSend 'params[0].{#this.username}' -b -x 2
Press Q or Ctrl+C to abort.
Affect(class-cnt:2 , method-cnt:1) cost in 28 ms.
ts=2019-09-26 15:19:49; [cost=0.006187ms] result=@ArrayList[
@String[u0],
@String[u1],
@String[u2],
@String[u3],
@String[u4],
@String[u5],
@String[u6],
@String[u7],
@String[u8],
@String[u9],
]
watch com.example.httpclientdemo.HttpclientDemoApplication doSend 'params[0].{? #this.id > 8}' -b -x 2
Press Q or Ctrl+C to abort.
Affect(class-cnt:2 , method-cnt:1) cost in 33 ms.
ts=2019-09-26 15:23:10; [cost=0.006812ms] result=@ArrayList[
@User[
username=@String[u9],
id=@Long[9],
],
]
在投影中过滤后计数
watch com.example.HttpclientDemoApplication doSend 'params[0].{? #this.username.endsWith("9")}.size()'
选择第一个匹配项
watch com.example.HttpclientDemoApplication doSend 'params[0].{^#this.username.startsWith("u")}' -b -x 2
选择最后一个匹配项
watch com.example.HttpclientDemoApplication doSend 'params[0].{$#this.username.startsWith("u")}' -b -x 2
10、条件表达式和观察表达式
ognl有两种表达式,条件表达式和观察表达式,条件表达式关注于这次请求能不能拦截到,观察表达式关注于这次请求打印出什么东西,两者区别如下:
条件表达式过滤的是一次调用,判断该次调用能否返回
观察表达式里的过滤,过滤的是该次调用的数据,不管怎么写,该次调用一定返回
案例如下:
watch com.example.HttpclientDemoApplication doSend
'params[0].{#this.username}' 'params[0].{? #this.id>7}.size()>0'
其中'params[0].{#this.username}’是观察表达式,'params[0].{? #this.id>7}.size()>0’是条件表达式。
观察表达式和条件表达式都可以用ognl。文章来源:https://www.toymoban.com/news/detail-431635.html
参考
watch | arthas文章来源地址https://www.toymoban.com/news/detail-431635.html
到了这里,关于arthas--watch函数执行数据观测的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!