理解字节码java

时间:2012-12-18 10:15:34

标签: java exception jvm bytecode

是否可以从java字节码中读取或识别INVOKESPECIAL指令引用的类?如果有,怎么样?另外,我如何知道跳转后执行的下一行是什么?

请记住,我想制作一个执行此操作的程序。我想要做的是找到一种方法来自动通过字节码本地化异常处理。

2 个答案:

答案 0 :(得分:1)

您可以查看有关http://asm.ow2.org/的此框架。 “ASM框架是执行字节码操作的最快,最灵活和众所周知的框架”

答案 1 :(得分:0)

有很多框架用于字节码操作there。但我个人更喜欢ASM。像机制这样的XML解析更容易学习。

例如,您可以使用此代码列出jar文件中的所有INVOKESPECIAL次调用: 它会打印这样的行:

INVOKESPECIAL[ opcode=183, owner=java/lang/StringBuilder, name=<init>, desc=()V]

您可以说<init> java/lang/StringBuilder的{​​{1}}函数由INVOKESPECIAL引用。

JarFile jarFile = new JarFile("xxx.jar");
Enumeration<JarEntry> entries = jarFile.entries();

while (entries.hasMoreElements()) {
    JarEntry jarEntry = entries.nextElement();
    if (jarEntry != null && jarEntry.getName().endsWith(".class")) {
        InputStream eis = jarFile.getInputStream(jarEntry);
        ClassReader classReader = new ClassReader(eis);
        ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
        MyClassVisitor mcw = new MyClassVisitor(Opcodes.ASM4, cw);
        classReader.accept(mcw, 0);
        eis.close();
    }
}


class MyClassVisitor extends ClassVisitor {
private int api;
public MyClassVisitor(int api, ClassWriter cw) {
    super(api, cw);
    this.api = api;
}
@Override
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
    MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);
    return new MyMethodVisitor(api, mv);
}
class MyMethodVisitor extends MethodVisitor {

    public MyMethodVisitor(int api, MethodVisitor mv) {
        super(api, mv);
    }
    @Override
    public void visitMethodInsn(int opcode, String owner, String name, String desc) {
        if (opcode == Opcodes.INVOKESPECIAL) {
            System.out.println("INVOKESPECIAL[ opcode=" + opcode + ", owner=" + owner + ", name=" + name
                    + ", desc=" + desc+"]");
        }
        super.visitMethodInsn(opcode, owner, name, desc);
    }
}
}