@toc
提示:
jad
命令将 JVM 中实际运行的 class 的 byte code 反编译成 java 代码,便于你理解业务逻辑
参数说明:
参数名称 | 参数说明 |
---|---|
class-pattern | 类名表达式匹配 |
| 类所属 ClassLoader 的 hashcode |
| 指定执行表达式的 ClassLoader 的 class name |
E | 开启正则表达式匹配,默认为通配符匹配 |
基本用法
: jad 全路径类名 方法名
[arthas@24056]$ jad com.hero.lte.ems.sysmanager.cache.SMTaskCache executeTask
ClassLoader:
+-sun.misc.Launcher$AppClassLoader@18b4aac2
+-sun.misc.Launcher$ExtClassLoader@232204a1
Location:
/home/ems/ems_eam/app/ems-eam-sysmanager-server-1.0-SNAPSHOT.jar
/*
* WARNING - Removed try catching itself - possible behaviour change.
*/
public static void executeTask(Long sysTaskId, Date nextTime) {
try {
/*233*/ lock.writeLock().lock();
/*234*/ Long curTime = System.currentTimeMillis();
/*235*/ for (SmconfigDmTaskMain smconfigdmtaskmain : taskCache) {
/*237*/ if (!smconfigdmtaskmain.getSysTaskId().equals(sysTaskId)) continue;
/*240*/ if (smconfigdmtaskmain.getSysTaskEndTime() != null && smconfigdmtaskmain.getSysTaskEndTime().getTime() <= curTime) {
/*241*/ smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.PASTDUE.getCodeString());
/*242*/ smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.INVALID.getCode());
} else {
/*244*/ smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.EXECUTING.getCodeString());
/*245*/ smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.ACTIVE.getCode());
}
/*247*/ smconfigdmtaskmain.setSysTaskNextTime(nextTime);
}
}
finally {
/*253*/ lock.writeLock().unlock();
}
}
Affect(row-cnt:1) cost in 1003 ms.
[arthas@24056]$
默认情况下,反编译结果里会带有ClassLoader
信息,通过--source-only
选项,可以只打印源代码。方便和mc/retransform命令结合使用。
基本用法
: jad --source-only 全路径类名 方法名
[arthas@24056]$ jad --source-only com.hero.lte.ems.sysmanager.cache.SMTaskCache executeTask
/*
* WARNING - Removed try catching itself - possible behaviour change.
*/
public static void executeTask(Long sysTaskId, Date nextTime) {
try {
/*233*/ lock.writeLock().lock();
/*234*/ Long curTime = System.currentTimeMillis();
/*235*/ for (SmconfigDmTaskMain smconfigdmtaskmain : taskCache) {
/*237*/ if (!smconfigdmtaskmain.getSysTaskId().equals(sysTaskId)) continue;
/*240*/ if (smconfigdmtaskmain.getSysTaskEndTime() != null && smconfigdmtaskmain.getSysTaskEndTime().getTime() <= curTime) {
/*241*/ smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.PASTDUE.getCodeString());
/*242*/ smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.INVALID.getCode());
} else {
/*244*/ smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.EXECUTING.getCodeString());
/*245*/ smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.ACTIVE.getCode());
}
/*247*/ smconfigdmtaskmain.setSysTaskNextTime(nextTime);
}
}
finally {
/*253*/ lock.writeLock().unlock();
}
}
[arthas@24056]$
基本用法
: jad --source-only 全路径类名 方法名 --lineNumber false
--lineNumber
参数默认值为 true,显示指定为 false 则不打印行号。
[arthas@24056]$ jad --source-only com.hero.lte.ems.sysmanager.cache.SMTaskCache executeTask --lineNumber false
/*
* WARNING - Removed try catching itself - possible behaviour change.
*/
public static void executeTask(Long sysTaskId, Date nextTime) {
try {
lock.writeLock().lock();
Long curTime = System.currentTimeMillis();
for (SmconfigDmTaskMain smconfigdmtaskmain : taskCache) {
if (!smconfigdmtaskmain.getSysTaskId().equals(sysTaskId)) continue;
if (smconfigdmtaskmain.getSysTaskEndTime() != null && smconfigdmtaskmain.getSysTaskEndTime().getTime() <= curTime) {
smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.PASTDUE.getCodeString());
smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.INVALID.getCode());
} else {
smconfigdmtaskmain.setSysTaskState(SMTaskStatusEnum.EXECUTING.getCodeString());
smconfigdmtaskmain.setTaskSwitch(SMTaskSwitchEnum.ACTIVE.getCode());
}
smconfigdmtaskmain.setSysTaskNextTime(nextTime);
}
}
finally {
lock.writeLock().unlock();
}
}
[arthas@24056]$
提示:
当有多个
ClassLoader
都加载了这个类时,jad
命令会输出对应ClassLoader
实例的hashcode
,然后你只需要重新执行jad
命令,并使用参数-c <hashcode>
就可以反编译指定 ClassLoader 加载的那个类了;
基本用法
: jad 全路径类名
$ jad org.apache.log4j.Logger
Found more than one class for: org.apache.log4j.Logger, Please use jad -c hashcode org.apache.log4j.Logger
HASHCODE CLASSLOADER
69dcaba4 +-monitor's ModuleClassLoader
6e51ad67 +-java.net.URLClassLoader@6e51ad67
+-sun.misc.Launcher$AppClassLoader@6951a712
+-sun.misc.Launcher$ExtClassLoader@6fafc4c2
2bdd9114 +-pandora-qos-service's ModuleClassLoader
4c0df5f8 +-pandora-framework's ModuleClassLoader
Affect(row-cnt:0) cost in 38 ms.
$ jad org.apache.log4j.Logger -c 69dcaba4
ClassLoader:
+-monitor's ModuleClassLoader
Location:
/Users/admin/app/log4j-1.2.14.jar
package org.apache.log4j;
import org.apache.log4j.spi.*;
public class Logger extends Category
{
private static final String FQCN;
protected Logger(String name)
{
super(name);
}
...
Affect(row-cnt:1) cost in 190 ms.
对于只有唯一实例的 ClassLoader 还可以通过--classLoaderClass
指定 class name,使用起来更加方便:
--classLoaderClass
的值是 ClassLoader 的类名,只有匹配到唯一的 ClassLoader 实例时才能工作,目的是方便输入通用命令,而-c <hashcode>
是动态变化的。
2.Arthas dashboard(当前系统的实时数据面板)
3.Arthas thread(查看当前JVM的线程堆栈信息)
5.Arthas sysprop(查看和修改JVM的系统属性)
7.Arthas vmoption(查看和修改 JVM里诊断相关的option)
9.Arthas heapdump(dump java heap, 类似 jmap 命令的 heap dump 功能)
10.Arthas logger(查看 logger 信息,更新 logger level)
14.Arthas perfcounter(查看当前 JVM 的 Perf Counter 信息)
15.Arthas vmtool(从 jvm 里查询对象,执行 forceGc)
17.Arthas mc(Memory Compiler/内存编译器 )
18.Arthas redefine(加载外部的.class文件,redefine到JVM里 )
19.Arthas classloader (查看 classloader 的继承树,urls,类加载信息)
23.Arthas stack (输出当前方法被调用的调用路径)
24.Arthas trace (方法内部调用路径,并输出方法路径上的每个节点上耗时)
25.Arthas tt(方法执行数据的时空隧道,记录下指定方法每次调用的入参和返回信息,并能对这些不同的时间下调用进行观测)
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。