Получить данные о работе с помощью javassist - PullRequest
0 голосов
/ 28 февраля 2012

Я пытаюсь проанализировать байт-код с помощью 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, ни пока что не дали положительного результата.

1 Ответ

1 голос
/ 29 февраля 2012

Это тяжелая работа, требующая понимания формата класса JVM и набора инструкций. Любой ответ может дать только частичные данные. Особенно с Javassist . В инструкции обычно один параметр - это индекс в пуле констант, где есть объект с индексом для имени и индексом для типа. Так что знайте, во что вы ввязываетесь.

Если класс скомпилирован с отладочной информацией, имена приватных полей доступны (ClassFile.getFields).

Есть хороший инструмент Dump, заполненный 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 });
}
...