使用javassist获取操作数据

时间:2012-02-28 19:07:56

标签: java bytecode-manipulation javassist

我正在尝试使用简单的类MyData

使用javassist分析字节代码
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文档都没有给出任何积极的结果。

1 个答案:

答案 0 :(得分:1)

这很难,需要了解JVM类格式和指令集。 任何答案都只能提供部分数据。尤其是Javassist。在指令中,通常一个参数是常量池中的索引,其中存在具有名称索引的对象和类型的索引。所以要注意你要进入的目标。

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

有一个很好的转储工具,包含了javassist。

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 });
}