0

我正在尝试使用简单的类使用 javassist 分析字节码MyData

class MyData {
    private Collection<String> strings = new ArrayList<String>();

        // .....................    
    public void add(String str) {
        strings.add(str); // line number 35
    }
        // .....................    
}

这是我针对此类运行的代码:

ClassFile cf = new ClassFile(new DataInputStream(TryJavassist.class.getResourceAsStream("MyData.class")));
MethodInfo minfo = cf.getMethod("add");  
CodeAttribute ca = minfo.getCodeAttribute();
for (CodeIterator ci = ca.iterator(); ci.hasNext();) {
    int index = ci.next();
    int op = ci.byteAt(index);

    System.out.println(op + "=" + Mnemonic.OPCODE[op] + ": " + minfo.getLineNumber(index));
}

它可以工作并打印:

42=aload_0: 35
180=getfield: 35
43=aload_1: 35
185=invokeinterface: 35
87=pop: 35
177=return: 36

第 35 行调用add()名为 的集合的方法strings。我发布的代码片段仅检索invokeinterface第 35 行。好的,我可以知道它是类字段 ( getfield)。

我想知道如何获取其余信息:

  • 字段名称是strings
  • 被调用的接口方法是add()

到目前为止,谷歌搜索和阅读 API 文档都没有给出任何积极的结果。

4

1 回答 1

2

这是一项艰苦的工作,需要了解 JVM 类格式和指令集。任何答案只能给出部分数据。特别是使用Javassist。在指令中,一个参数通常是常量池中的一个索引,其中有一个对象具有一个名称的索引和一个类型的索引。所以要注意你正在进入什么。

如果使用调试信息编译类,则可以访问私有字段名称(ClassFile.getFields)。

有一个很好的带有 javassist 的 Dump 工具。

private void dumpMyDataClass() throws IOException, BadBytecode, Exception {
    ClassFile cf = new ClassFile(new DataInputStream(getClass().getResourceAsStream("MyData.class")));

    // Dump fields:
    for (Object fieldInfoObj : cf.getFields()) {
        FieldInfo fieldInfo = (FieldInfo) fieldInfoObj;
        System.out.printf("Field %s; %s%n", fieldInfo.getName(), fieldInfo.getDescriptor());
    }

    MethodInfo minfo = cf.getMethod("add");
    CodeAttribute ca = minfo.getCodeAttribute();
    for (CodeIterator ci = ca.iterator(); ci.hasNext();) {
        int address = ci.next();
        int op = ci.byteAt(address);

        String params = "";
        switch (op) {
            case Opcode.INVOKEINTERFACE:
                int a1 = ci.s16bitAt(address + 1);
                params += " " + cf.getConstPool().getInterfaceMethodrefName(a1);
                System.out.println("a1 = " + a1);
                break;
        }

        System.out.printf("Line %4d. Address %7d: %s%s%n", minfo.getLineNumber(address), address, Mnemonic.OPCODE[op], params);
    }

    // Command line tool of javassist:
    String pathToClass = System.getProperty("user.dir") + "/target/classes/jeggen/test2/MyData.class";
    Dump.main(new String[] { pathToClass });
}
于 2012-02-28T20:32:04.877 回答