Alibaba Arthas 使用学习之路,最详细讲解

1.Arthas 是什么?

   Arthas 是Alibaba 开发得一款java诊断工具,可以帮我们查看jvm使用情况,查看线程数,排除阻塞线程;监听类和方法得调用次数,失败次数(失败率),执行时常;监听方法得入参,出参,异常信息;快速反编译查看源码,并修改源码,编译运行;查看堆栈信息等。

2.这里列出一些常用命令并一一讲解

   最常用得命令:dashboard,thread,jvm,sc,sm,jad,mc,redefine,monitor,watch。官方地址:https://arthas.aliyun.com/

dashboard

dashboard

   dashboard 当前系统的实时数据面板,按 ctrl+c 退出,当运行在Ali-tomcat时,会显示当前tomcat的实时信息,如HTTP请求的qps, rt, 错误数, 线程池信息等等。
   数据说明如下:

  • ID: Java级别的线程ID,注意这个ID不能跟jstack中的nativeID一一对应
  • NAME: 线程名
  • GROUP: 线程组名
  • PRIORITY: 线程优先级, 1~10之间的数字,越大表示优先级越高
  • STATE: 线程的状态
  • CPU%: 线程消耗的cpu占比,采样100ms,将所有线程在这100ms内的cpu使用量求和,再算出每个线程的cpu使用占比。
  • TIME: 线程运行总时间,数据格式为 分:秒
  • INTERRUPTED: 线程当前的中断位状态
  • DAEMON: 是否是守护线程

thread

   查看当前线程信息,查看线程的堆栈
   参数说明:id(线程id),[n:](指定最忙的前N个线程并打印堆栈),[b](找出当前阻塞其他线程的线程),[i<value>](指定cpu占比统计的采样间隔,单位为毫秒,建议5000)
   示例: thread -n 3 找出前3个线程的堆栈信息

$ thread -n 3
"as-command-execute-daemon" Id=58 cpuUsage=76% RUNNABLE
    at java.management@12.0.1/sun.management.ThreadImpl.dumpThreads0(Native Method)
    at java.management@12.0.1/sun.management.ThreadImpl.getThreadInfo(ThreadImpl.java:466)
    at com.taobao.arthas.core.command.monitor200.ThreadCommand.processTopBusyThreads(ThreadCommand.java:133)
    at com.taobao.arthas.core.command.monitor200.ThreadCommand.process(ThreadCommand.java:79)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl.process(AnnotatedCommandImpl.java:82)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl.access$100(AnnotatedCommandImpl.java:18)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl$ProcessHandler.handle(AnnotatedCommandImpl.java:111)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl$ProcessHandler.handle(AnnotatedCommandImpl.java:108)
    at com.taobao.arthas.core.shell.system.impl.ProcessImpl$CommandProcessTask.run(ProcessImpl.java:370)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)

    Number of locked synchronizers = 1
    - java.util.concurrent.ThreadPoolExecutor$Worker@7808db3c


"NioBlockingSelector.BlockPoller-1" Id=24 cpuUsage=17% RUNNABLE (in native)
    at java.base@12.0.1/sun.nio.ch.EPoll.wait(Native Method)
    at java.base@12.0.1/sun.nio.ch.EPollSelectorImpl.doSelect(EPollSelectorImpl.java:120)
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.lockAndDoSelect(SelectorImpl.java:124)
    -  locked sun.nio.ch.Util$2@1a574b17
    -  locked sun.nio.ch.EPollSelectorImpl@7b0d44f5
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.select(SelectorImpl.java:136)
    at org.apache.tomcat.util.net.NioBlockingSelector$BlockPoller.run(NioBlockingSelector.java:304)


"http-nio-8087-ClientPoller-1" Id=36 cpuUsage=5% RUNNABLE (in native)
    at java.base@12.0.1/sun.nio.ch.EPoll.wait(Native Method)
    at java.base@12.0.1/sun.nio.ch.EPollSelectorImpl.doSelect(EPollSelectorImpl.java:120)
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.lockAndDoSelect(SelectorImpl.java:124)
    -  locked sun.nio.ch.Util$2@703ba4c0
    -  locked sun.nio.ch.EPollSelectorImpl@6a6a92bd
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.select(SelectorImpl.java:136)
    at org.apache.tomcat.util.net.NioEndpoint$Poller.run(NioEndpoint.java:743)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)


Affect(row-cnt:0) cost in 131 ms.

   示例:thread 打出所有运行线程

$ thread
Threads Total: 37, NEW: 0, RUNNABLE: 14, BLOCKED: 0, WAITING: 17, TIMED_WAITING: 6, TERMINATED: 0                                                                                                                                       
ID                 NAME                                                      GROUP                                  PRIORITY           STATE               %CPU               TIME               INTERRUPTED         DAEMON             
59                 as-command-execute-daemon                                 system                                 10                 RUNNABLE            100                0:0                false               true               
42                 AsyncAppender-Worker-arthas-cache.result.AsyncAppender    system                                 9                  WAITING             0                  0:0                false               true               
40                 Attach Listener                                           system                                 9                  RUNNABLE            0                  0:0                false               true               
14                 Catalina-utility-1                                        main                                   1                  WAITING             0                  0:1                false               false              
15                 Catalina-utility-2                                        main                                   1                  TIMED_WAITING       0                  0:0                false               false              
10                 Common-Cleaner                                            InnocuousThreadGroup                   8                  TIMED_WAITING       0                  0:0                false               true               
39                 DestroyJavaVM                                             main                                   5                  RUNNABLE            0                  0:12               false               false              
3                  Finalizer                                                 system                                 8                  WAITING             0                  0:0                false               true               
19                 MQTT Call: mqttId_inbound                                 main                                   5                  WAITING             0                  0:0                false               false              
23                 MQTT Ping: mqttId_inbound                                 main                                   5                  TIMED_WAITING       0                  0:0                false               false              
20                 MQTT Rec: mqttId_inbound                                  main                                   5                  RUNNABLE            0                  0:0                false               false              
21                 MQTT Snd: mqttId_inbound                                  main                                   5                  WAITING             0                  0:0                false               false              
24                 NioBlockingSelector.BlockPoller-1                         main                                   5                  RUNNABLE            0                  0:0                false               true               
2                  Reference Handler                                         system                                 10                 RUNNABLE            0                  0:0                false               true               
4                  Signal Dispatcher                                         system                                 9                  RUNNABLE            0                  0:0                false               true               
16                 container-0                                               main                                   5                  TIMED_WAITING       0                  0:0                false               false              
37                 http-nio-8087-Acceptor-0                                  main                                   5                  RUNNABLE            0                  0:0                false               true               
35                 http-nio-8087-ClientPoller-0                              main                                   5                  RUNNABLE            0                  0:0                false               true               
36                 http-nio-8087-ClientPoller-1                              main                                   5                  RUNNABLE            0                  0:0                false               true               
25                 http-nio-8087-exec-1                                      main                                   5                  WAITING             0                  0:0                false               true               
34                 http-nio-8087-exec-10                                     main                                   5                  WAITING             0                  0:0                false               true               
26                 http-nio-8087-exec-2                                      main                                   5                  WAITING             0                  0:0                false               true               
27                 http-nio-8087-exec-3                                      main                                   5                  WAITING             0                  0:0                false               true               
28                 http-nio-8087-exec-4                                      main                                   5                  WAITING             0                  0:0                false               true               
29                 http-nio-8087-exec-5                                      main                                   5                  WAITING             0                  0:0                false               true               
30                 http-nio-8087-exec-6                                      main                                   5                  WAITING             0                  0:0                false               true               
31                 http-nio-8087-exec-7                                      main                                   5                  WAITING             0                  0:0                false               true               
32                 http-nio-8087-exec-8                                      main                                   5                  WAITING             0                  0:0                false               true               
33                 http-nio-8087-exec-9                                      main                                   5                  WAITING             0                  0:0                false               true               
44                 job-timeout                                               system                                 9                  TIMED_WAITING       0                  0:0                false               true               
45                 nioEventLoopGroup-2-1                                     system                                 10                 RUNNABLE            0                  0:0                false               false              
49                 nioEventLoopGroup-2-2                                     system                                 10                 RUNNABLE            0                  0:1                false               false              
53                 nioEventLoopGroup-2-3                                     system                                 10                 RUNNABLE            0                  0:0                false               false              
46                 nioEventLoopGroup-3-1                                     system                                 10                 RUNNABLE            0                  0:0                false               false              
22                 pool-2-thread-4                                           main                                   5                  WAITING             0                  0:0                false               false              
47                 pool-3-thread-1                                           system                                 5                  TIMED_WAITING       0                  0:0                false               false              
48                 pool-4-thread-1                                           system                                 5                  WAITING             0                  0:0                false               false              
Affect(row-cnt:0) cost in 113 ms.

   示例:thread 20 查看线程id为20的堆栈信息

$ thread 20
"MQTT Rec: mqttId_inbound" Id=20 RUNNABLE (in native)
    at java.base@12.0.1/java.net.SocketInputStream.socketRead0(Native Method)
    at java.base@12.0.1/java.net.SocketInputStream.socketRead(SocketInputStream.java:115)
    at java.base@12.0.1/java.net.SocketInputStream.read(SocketInputStream.java:168)
    at java.base@12.0.1/java.net.SocketInputStream.read(SocketInputStream.java:140)
    at java.base@12.0.1/java.net.SocketInputStream.read(SocketInputStream.java:200)
    at java.base@12.0.1/java.io.DataInputStream.readByte(DataInputStream.java:270)
    at org.eclipse.paho.client.mqttv3.internal.wire.MqttInputStream.readMqttWireMessage(MqttInputStream.java:92)
    at org.eclipse.paho.client.mqttv3.internal.CommsReceiver.run(CommsReceiver.java:133)
    at java.base@12.0.1/java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:515)
    at java.base@12.0.1/java.util.concurrent.FutureTask.run(FutureTask.java:264)
    at java.base@12.0.1/java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:304)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)

    Number of locked synchronizers = 1
    - java.util.concurrent.ThreadPoolExecutor$Worker@79cf8c25

Affect(row-cnt:0) cost in 33 ms.

   示例:thread -b 找出阻塞其他线程的线程

$ thread -b
No most blocking thread found!
Affect(row-cnt:0) cost in 43 ms.

   示例:thread -n 3 -i 5000 阻塞指定5秒后收到前3线程的堆栈信息

$ thread -n 3  -i 5000
"Catalina-utility-1" Id=14 cpuUsage=21% TIMED_WAITING on java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject@72b49c8
    at java.base@12.0.1/jdk.internal.misc.Unsafe.park(Native Method)
    -  waiting on java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject@72b49c8
    at java.base@12.0.1/java.util.concurrent.locks.LockSupport.parkNanos(LockSupport.java:235)
    at java.base@12.0.1/java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.awaitNanos(AbstractQueuedSynchronizer.java:2123)
    at java.base@12.0.1/java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(ScheduledThreadPoolExecutor.java:1182)
    at java.base@12.0.1/java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(ScheduledThreadPoolExecutor.java:899)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor.getTask(ThreadPoolExecutor.java:1054)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1114)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
    at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)


"http-nio-8087-ClientPoller-1" Id=36 cpuUsage=14% RUNNABLE (in native)
    at java.base@12.0.1/sun.nio.ch.EPoll.wait(Native Method)
    at java.base@12.0.1/sun.nio.ch.EPollSelectorImpl.doSelect(EPollSelectorImpl.java:120)
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.lockAndDoSelect(SelectorImpl.java:124)
    -  locked sun.nio.ch.Util$2@703ba4c0
    -  locked sun.nio.ch.EPollSelectorImpl@6a6a92bd
    at java.base@12.0.1/sun.nio.ch.SelectorImpl.select(SelectorImpl.java:136)
    at org.apache.tomcat.util.net.NioEndpoint$Poller.run(NioEndpoint.java:743)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)


"as-command-execute-daemon" Id=62 cpuUsage=10% RUNNABLE
    at java.management@12.0.1/sun.management.ThreadImpl.dumpThreads0(Native Method)
    at java.management@12.0.1/sun.management.ThreadImpl.getThreadInfo(ThreadImpl.java:466)
    at com.taobao.arthas.core.command.monitor200.ThreadCommand.processTopBusyThreads(ThreadCommand.java:133)
    at com.taobao.arthas.core.command.monitor200.ThreadCommand.process(ThreadCommand.java:79)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl.process(AnnotatedCommandImpl.java:82)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl.access$100(AnnotatedCommandImpl.java:18)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl$ProcessHandler.handle(AnnotatedCommandImpl.java:111)
    at com.taobao.arthas.core.shell.command.impl.AnnotatedCommandImpl$ProcessHandler.handle(AnnotatedCommandImpl.java:108)
    at com.taobao.arthas.core.shell.system.impl.ProcessImpl$CommandProcessTask.run(ProcessImpl.java:370)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
    at java.base@12.0.1/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
    at java.base@12.0.1/java.lang.Thread.run(Thread.java:835)

    Number of locked synchronizers = 1
    - java.util.concurrent.ThreadPoolExecutor$Worker@2d2909be


Affect(row-cnt:0) cost in 5045 ms.

jvm

   查看当前JVM信息,这里只拿部分信息来说

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 THREAD                                                                                                                                                                                                                                 
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 COUNT                                                             37                                                                                                                                                                   
 DAEMON-COUNT                                                      22                                                                                                                                                                   
 PEAK-COUNT                                                        38                                                                                                                                                                   
 STARTED-COUNT                                                     55                                                                                                                                                                   
 DEADLOCK-COUNT                                                    0                                                                                                                                                                    
                                                                                                                                                                                                                                        
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 FILE-DESCRIPTOR                                                                                                                                                                                                                        
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 MAX-FILE-DESCRIPTOR-COUNT                                         -1                                                                                                                                                                   
 OPEN-FILE-DESCRIPTOR-COUNT                                        -1         

   THREAD 相关

  • COUNT: JVM当前活跃的线程数
  • DAEMON-COUNT: JVM当前活跃的守护线程数
  • PEAK-COUNT: 从JVM启动开始曾经活着的最大线程数
  • STARTED-COUNT: 从JVM启动开始总共启动过的线程次数
  • DEADLOCK-COUNT: JVM当前死锁的线程数

   FILE-DESCRIPTOR 相关

  • MAX-FILE-DESCRIPTOR-COUNT:JVM进程最大可以打开的文件描述符数
  • OPEN-FILE-DESCRIPTOR-COUNT:JVM当前打开的文件描述符数

sc

   查看JVM已加载的类信息,
   参数说明:lass(包名+类名),method(方法名),[d](输出详细信息),[E](开启正则表达式匹配,默认为通配符匹配),[f](输出当前类的成员变量信息(需要配合参数-d一起使用)),[x:](指定输出静态变量时属性的遍历深度,默认为 0,即直接使用 toString 输出)
   class-pattern支持全限定名,如com.taobao.test.AAA,也支持com/taobao/test/AAA这样的格式,这样,我们从异常堆栈里面把类名拷贝过来的时候,不需要在手动把/替换为 . 啦。
   示例:sc com.cloud*

$ sc com.cloud*
com.cloud.mqtt.MqttApplication
com.cloud.mqtt.MqttApplication$$EnhancerBySpringCGLIB$$b4fbeb8a
com.cloud.mqtt.mqtt.MqttConfig
com.cloud.mqtt.mqtt.MqttConfig$$EnhancerBySpringCGLIB$$d1953d4e
com.cloud.mqtt.mqtt.MqttConfig$$EnhancerBySpringCGLIB$$d1953d4e$$FastClassBySpringCGLIB$$69e081d8
com.cloud.mqtt.mqtt.MqttConfig$$FastClassBySpringCGLIB$$bf72748c
com.cloud.mqtt.mqtt.MqttConfig$1
com.cloud.mqtt.mqtt.MqttReceiveConfig
com.cloud.mqtt.mqtt.MqttReceiveConfig$$EnhancerBySpringCGLIB$$5631a71d
com.cloud.mqtt.mqtt.MqttSenderConfig
com.cloud.mqtt.mqtt.MqttSenderConfig$$EnhancerBySpringCGLIB$$3bfb2a03
com.cloud.mqtt.mqtt.Publisher
com.cloud.mqtt.web.MqttGateway
com.cloud.mqtt.web.TestController
com.sun.proxy.$Proxy65
Affect(row-cnt:15) cost in 20 ms.

   示例:sc -d -f com.cloud.mqtt.web.TestController 打印类的信息以及字段的信息

$ sc -d -f com.cloud.mqtt.web.TestController
 class-info        com.cloud.mqtt.web.TestController                                                                                                                                                                                    
 code-source       file:/home/fengqianrun/opt/server/mqtt-0.0.1.jar!/BOOT-INF/classes!/                                                                                                                                                 
 name              com.cloud.mqtt.web.TestController                                                                                                                                                                                    
 isInterface       false                                                                                                                                                                                                                
 isAnnotation      false                                                                                                                                                                                                                
 isEnum            false                                                                                                                                                                                                                
 isAnonymousClass  false                                                                                                                                                                                                                
 isArray           false                                                                                                                                                                                                                
 isLocalClass      false                                                                                                                                                                                                                
 isMemberClass     false                                                                                                                                                                                                                
 isPrimitive       false                                                                                                                                                                                                                
 isSynthetic       false                                                                                                                                                                                                                
 simple-name       TestController                                                                                                                                                                                                       
 modifier          public                                                                                                                                                                                                               
 annotation        org.springframework.web.bind.annotation.RestController,org.springframework.web.bind.annotation.RequestMapping                                                                                                        
 interfaces                                                                                                                                                                                                                             
 super-class       +-java.lang.Object                                                                                                                                                                                                   
 class-loader      +-org.springframework.boot.loader.LaunchedURLClassLoader@439f5b3d                                                                                                                                                    
                     +-jdk.internal.loader.ClassLoaders$AppClassLoader@5bc2b487                                                                                                                                                         
                       +-jdk.internal.loader.ClassLoaders$PlatformClassLoader@75d2da2d                                                                                                                                                  
 classLoaderHash   439f5b3d                                                                                                                                                                                                             
 fields            modifier  private                                                                                                                                                                                                    
                   type      com.cloud.mqtt.web.MqttGateway                                                                                                                                                                                                      
                   name      mqttGateway                                                                                                                                                                                                
                   annotationjavax.annotation.Resource                                                                                                                                                                                  
                                                                                                                                                                                                                                        

Affect(row-cnt:1) cost in 15 ms.

sm

   查看已加载类的方法信息,命令只能看到由当前类所声明 (declaring) 的方法,父类则无法看到。
   参数说明:class(包名+类名),method(方法名称),[d](展示每个方法的详细信息),[E](开启正则表达式匹配,默认为通配符匹配)
   示例:sm com.cloud.mqtt.web.TestController 查询TestController下的所有方法

$ sm  com.cloud.mqtt.web.TestController
com.cloud.mqtt.web.TestController <init>()V
com.cloud.mqtt.web.TestController test(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
com.cloud.mqtt.web.TestController sendMqtt(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
com.cloud.mqtt.web.TestController sendTest(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
Affect(row-cnt:4) cost in 12 ms.

   示例:sm -d com.cloud.mqtt.web.TestController test 查询TestController下的test方法详情

$ sm -d com.cloud.mqtt.web.TestController test
 declaring-class  com.cloud.mqtt.web.TestController                                                                                                                                                                                     
 method-name      test                                                                                                                                                                                                                  
 modifier         public                                                                                                                                                                                                                
 annotation       org.springframework.web.bind.annotation.GetMapping                                                                                                                                                                    
 parameters       java.lang.String                                                                                                                                                                                                      
                  java.lang.String                                                                                                                                                                                                      
 return           java.lang.String                                                                                                                                                                                                      
 exceptions                                                                                                                                                                                                                             

Affect(row-cnt:1) cost in 13 ms.

jad

   jad 反编译指定已加载类得源码,jad 在Arthas Console上,反编译出来得源码是带语法高亮得,阅读方便,当然反编译出来得java代码可能会存在语法错误,但不影响你的进行阅读。
   参数说明:class(包名+类名),[c:](类所属ClassLoader得hashcode),[E](开启正则表达匹配,默认通配符)
   示例:反编译 com.cloud.mqtt.web.TestController

$ jad com.cloud.mqtt.web.TestController

ClassLoader:                                                                                                                                                                                                                            
+-org.springframework.boot.loader.LaunchedURLClassLoader@439f5b3d                                                                                                                                                                       
  +-jdk.internal.loader.ClassLoaders$AppClassLoader@5bc2b487                                                                                                                                                                            
    +-jdk.internal.loader.ClassLoaders$PlatformClassLoader@75d2da2d                                                                                                                                                                     

Location:                                                                                                                                                                                                                               
file:/home/fengqianrun/opt/server/mqtt-0.0.1.jar!/BOOT-INF/classes!/                                                                                                                                                                    

/*
 * Decompiled with CFR 0_132.
 * 
 * Could not load the following classes:
 *  com.cloud.mqtt.web.MqttGateway
 *  javax.annotation.Resource
 *  org.springframework.web.bind.annotation.GetMapping
 *  org.springframework.web.bind.annotation.RequestMapping
 *  org.springframework.web.bind.annotation.RequestParam
 *  org.springframework.web.bind.annotation.RestController
 */
package com.cloud.mqtt.web;

import com.cloud.mqtt.web.MqttGateway;
import javax.annotation.Resource;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping(value={"/test"})
public class TestController {
    @Resource
    private MqttGateway mqttGateway;

    @GetMapping(value={"/"})
    public String test(String a, String b) {
        String rt = a + b;
        return rt;
    }

    @GetMapping(value={"/send"})
    public String sendMqtt(@RequestParam(value="data") String data, @RequestParam(value="topic") String topic) {
        String rt = "sendMqtt";
        return rt;
    }

    @GetMapping(value={"/sendTest"})
    public String sendTest(@RequestParam(value="data") String data, @RequestParam(value="topic") String topic) {
        String rt = "sendMqtt";
        return "OK";
    }
}

Affect(row-cnt:1) cost in 394 ms.

   示例:反编译 com.cloud.mqtt.web.TestController test方法

$ jad com.cloud.mqtt.web.TestController test

ClassLoader:                                                                                                                                                                                                                            
+-org.springframework.boot.loader.LaunchedURLClassLoader@439f5b3d                                                                                                                                                                       
  +-jdk.internal.loader.ClassLoaders$AppClassLoader@5bc2b487                                                                                                                                                                            
    +-jdk.internal.loader.ClassLoaders$PlatformClassLoader@75d2da2d                                                                                                                                                                     

Location:                                                                                                                                                                                                                               
file:/home/fengqianrun/opt/server/mqtt-0.0.1.jar!/BOOT-INF/classes!/                                                                                                                                                                    

@GetMapping(value={"/"})
public String test(String a, String b) {
    String rt = a + b;
    return rt;
}

Affect(row-cnt:1) cost in 67 ms.

mc

   内存编译器,编译.java文件生成.class文件。编译生成.class之后们,可以节后 redefine 命令实现热更新代码。
   参数说明:[c:](指定classloader),[d](指定输出目录)

mc /tmp/Test.java 编译Test.java文件
mc -c 327a647b /tmp/Test.java 指定classloader编译
mc -d /tmp/output /tmp/ClassA.java /tmp/ClassB.java 编译到指定目录

redefine

   加载外部的 .class 文件到redefine jvm已加载的类。
   redefine后的原来的类不能恢复,redefine有可能失败(比如增加了新的field),参考jdk本身的文档。
   参数说明:[c:](ClassLoader的hashcode),[p:](外部的.class文件的完整路径,支持多个)

redefine /tmp/Test.class
结合 jad/mc 命令使用:
jad --source-only com.example.demo.arthas.user.UserController > /tmp/UserController.java
mc /tmp/UserController.java -d /tmp
redefine /tmp/com/example/demo/arthas/user/UserController.class
- jad命令反编译,然后可以用其它编译器,比如vim来修改源码
- mc命令来内存编译修改过的代码
- 用redefine命令加载新的字节码

   redefine的限制,不允许新增加field/method,正在跑的函数,没有退出不能生效,比如下面新增加的System.out.println,只有run()函数里的会生效。

monitor

   monitor 方法监控执行,是一个非实时得返回命令,意思就是可以一直监控某一个方法,直到你按 ctrl+c结束为止。
   监控得信息包括:timestamp(时间戳),class(java类),method(方法,包括构造方法和普通方法),total(调用次数),success(成功次数),fail(失败次数),rt(平均RT),fail-rate(失败率)。
   方法参数说明:class(包名+类名),method(方法名),[E](开启正则表达匹配,默认通配符),[c:](得到运行报告得间隔时间,默认是120秒)

$ monitor -c 5 com.cloud.mqtt.web.TestController test
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 60 ms.
 timestamp            class                              method  total  success  fail  avg-rt(ms)  fail-rate                                                                                                                            
-------------------------------------------------------------------------------------------------------------                                                                                                                           
 2019-08-19 14:29:03  com.cloud.mqtt.web.TestController  test    6      6        0     0.34        0.00%                                                                                                                                

 timestamp            class                              method  total  success  fail  avg-rt(ms)  fail-rate                                                                                                                            
-------------------------------------------------------------------------------------------------------------                                                                                                                           
 2019-08-19 14:29:08  com.cloud.mqtt.web.TestController  test    1      1        0     0.09        0.00%                                                                                                                                

 timestamp            class                              method  total  success  fail  avg-rt(ms)  fail-rate                                                                                                                            
-------------------------------------------------------------------------------------------------------------                                                                                                                           
 2019-08-19 14:29:13  com.cloud.mqtt.web.TestController  test    0      0        0     0.00        0.00%                                                                                                                                

 timestamp            class                              method  total  success  fail  avg-rt(ms)  fail-rate                                                                                                                            
-------------------------------------------------------------------------------------------------------------                                                                                                                           
 2019-08-19 14:29:18  com.cloud.mqtt.web.TestController  test    0      0        0     0.00        0.00%            

  我得类 com.cloud.mqtt.web.TestController ,方法为 test ,-c 5 得意思就是每5秒打印一次。

watch

   watch 方法执行数据观测。
   参数说明:class(类名表达式匹配),method(方法名表达式匹配),express(观察表达式),condition-express(条件表达式),[b](在方法调用之前观察),[e](在方法异常之后观察),[s](在方法返回之后观察),[f](在方法结束之后(正常返回和异常返回)观察),[E](开启正则表达式匹配,默认为通配符匹配),[x:](指定输出结果的属性遍历深度,默认为 1)
   特别说明:

  • watch 命令定义了4个观察事件点,即 -b 方法调用前,-e 方法异常后,-s 方法返回后,-f 方法结束后
  • 4个观察事件点 -b、-e、-s 默认关闭,-f 默认打开,当指定观察点被打开后,在相应事件点会对观察表达式进行求值并输出
  • 这里要注意方法入参和方法出参的区别,有可能在中间被修改导致前后不一致,除了 -b 事件点 params 代表方法入参外,其余事件都代表方法出参
  • 当使用 -b 时,由于观察事件点是在方法调用前,此时返回值或异常均不存在
curl "http://192.168.15.130:8087/test/?a=1&b=3" ,-x 2的意思是,是否深度解析参数,如果你的参数是一个对象,不深度解析,你看到的只是对象地址,深度解析的值,跟你参数所嵌套层数有关。
$ watch com.cloud.mqtt.web.TestController  test "{params,returnObj}" -x 2
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 41 ms.
ts=2019-08-19 17:02:09; [cost=0.241819ms] result=@ArrayList[
    @Object[][
        @String[1],
        @String[3],
    ],
    @String[13],
]

-b 意思是解析入参,-s解析出参, -n 2执行两次的意思,2次完则结束。结果的输出顺序和事件发生的先后顺序一致,和命令中 -s -b 的顺序无关。
$ watch com.cloud.mqtt.web.TestController  test "{params,returnObj}" -x 2 -b  -s -n 2
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 46 ms.
ts=2019-08-19 17:06:21; [cost=0.006941ms] result=@ArrayList[
    @Object[][
        @String[1],
        @String[3],
    ],
    null,
]
ts=2019-08-19 17:06:21; [cost=0.945661ms] result=@ArrayList[
    @Object[][
        @String[1],
        @String[3],
    ],
    @String[13],
]
Command execution times exceed limit: 2, so command will exit. You can set it with -n option.

条件表达式,第一个传参大于5才执行
$ watch com.cloud.mqtt.web.TestController  test "{params[0],returnObj}" "params[0] > 5"
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 39 ms.
ts=2019-08-19 17:21:58; [cost=0.183399ms] result=@ArrayList[
    @String[6],
    @String[63],
]

异常信息举例:-e 表示抛出异常时才触发,express中,表示异常信息的变量是throwExp
$ 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),
]

按照耗时进行过滤:#cost>200(单位是ms)表示只有当耗时大于200ms时才会输出,过滤掉执行时间小于200ms的调用
$ watch demo.MathGame primeFactors '{params, returnObj}' '#cost>200' -x 2
$ 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[2141897465],
    ],
    @ArrayList[
        @Integer[5],
        @Integer[428379493],
    ],
]

观察当前对象中的属性:如果想查看方法运行前后,当前对象中的属性,可以使用target关键字,代表当前对象,然后使用target.field_name访问当前对象的某个属性
$ watch com.cloud.mqtt.web.TestController  test 'target'
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 47 ms.
ts=2019-08-19 17:27:09; [cost=0.450697ms] result=@TestController[
    mqttGateway=@$Proxy65[gateway proxy for service interface [interface com.cloud.mqtt.web.MqttGateway]],
]

$ watch com.cloud.mqtt.web.TestController  test 'target.mqttGateway'
Press Q or Ctrl+C to abort.
Affect(class-cnt:1 , method-cnt:1) cost in 46 ms.
ts=2019-08-19 17:28:31; [cost=0.352704ms] result=@$Proxy65[
    m1=@Method[public boolean java.lang.Object.equals(java.lang.Object)],
    m10=@Method[public abstract boolean org.springframework.aop.framework.Advised.isExposeProxy()],
    m13=@Method[public abstract void org.springframework.aop.framework.Advised.addAdvisor(org.springframework.aop.Advisor) throws org.springframework.aop.framework.AopConfigException],
    m7=@Method[public abstract boolean org.springframework.aop.framework.Advised.isProxyTargetClass()],
    m15=@Method[public abstract void org.springframework.aop.framework.Advised.removeAdvisor(int) throws org.springframework.aop.framework.AopConfigException],
    m23=@Method[public abstract java.lang.Class[] org.springframework.aop.framework.Advised.getProxiedInterfaces()],
    m5=@Method[public abstract int org.springframework.aop.framework.Advised.indexOf(org.springframework.aop.Advisor)],
    m22=@Method[public abstract org.springframework.aop.TargetSource org.springframework.aop.framework.Advised.getTargetSource()],
    m18=@Method[public abstract void org.springframework.aop.framework.Advised.addAdvice(int,org.aopalliance.aop.Advice) throws org.springframework.aop.framework.AopConfigException],
    m19=@Method[public abstract void org.springframework.aop.framework.Advised.addAdvice(org.aopalliance.aop.Advice) throws org.springframework.aop.framework.AopConfigException],
    m0=@Method[public native int java.lang.Object.hashCode()],
    m24=@Method[public abstract boolean org.springframework.aop.framework.Advised.isInterfaceProxied(java.lang.Class)],
    m20=@Method[public abstract boolean org.springframework.aop.framework.Advised.removeAdvice(org.aopalliance.aop.Advice)],
    m9=@Method[public abstract void org.springframework.aop.framework.Advised.setExposeProxy(boolean)],
    m8=@Method[public abstract void org.springframework.aop.framework.Advised.setTargetSource(org.springframework.aop.TargetSource)],
    m2=@Method[public java.lang.String java.lang.Object.toString()],
    m26=@Method[public abstract java.lang.Class org.springframework.aop.TargetClassAware.getTargetClass()],
    m14=@Method[public abstract void org.springframework.aop.framework.Advised.addAdvisor(int,org.springframework.aop.Advisor) throws org.springframework.aop.framework.AopConfigException],
    m27=@Method[public abstract java.lang.Class org.springframework.core.DecoratingProxy.getDecoratedClass()],
    m16=@Method[public abstract boolean org.springframework.aop.framework.Advised.removeAdvisor(org.springframework.aop.Advisor)],
    m4=@Method[public abstract int org.springframework.aop.framework.Advised.indexOf(org.aopalliance.aop.Advice)],
    m6=@Method[public abstract boolean org.springframework.aop.framework.Advised.isFrozen()],
    m17=@Method[public abstract boolean org.springframework.aop.framework.Advised.replaceAdvisor(org.springframework.aop.Advisor,org.springframework.aop.Advisor) throws org.springframework.aop.framework.AopConfigException],
    m11=@Method[public abstract void org.springframework.aop.framework.Advised.setPreFiltered(boolean)],
    m3=@Method[public abstract void com.cloud.mqtt.web.MqttGateway.sendToMqtt(java.lang.String,java.lang.String)],
    m21=@Method[public abstract java.lang.String org.springframework.aop.framework.Advised.toProxyConfigString()],
    m25=@Method[public abstract org.springframework.aop.Advisor[] org.springframework.aop.framework.Advised.getAdvisors()],
    m12=@Method[public abstract boolean org.springframework.aop.framework.Advised.isPreFiltered()],
]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容

  • 这篇文章是我之前翻阅了不少的书籍以及从网络上收集的一些资料的整理,因此不免有一些不准确的地方,同时不同JDK版本的...
    高广超阅读 15,572评论 3 83
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 2,635评论 0 11
  • 【JAVA 线程】 线程 进程:是一个正在执行中的程序。每一个进程执行都有一个执行顺序。该顺序是一个执行路径,或者...
    Rtia阅读 2,764评论 2 20
  • 一、Python简介和环境搭建以及pip的安装 4课时实验课主要内容 【Python简介】: Python 是一个...
    _小老虎_阅读 5,734评论 0 10
  • 林炳文Evankaka原创作品。转载自http://blog.csdn.net/evankaka 本文主要讲了ja...
    ccq_inori阅读 648评论 0 4