首页 > 解决方案 > 如何复制 InsnList

问题描述

InsnList 没有复制自身的方法。

我试图迭代列表并将每个节点添加到新列表中。但是迭代列表的副本执行 npe

private static InsnList copy(InsnList insnList) {
        InsnList r = new InsnList();

        for (int i = 0; i < insnList.size(); i++)
            r.add(insnList.get(i));

        return r;
}
InsnList copy = copy(someList);

for (int i = 0; i < copy.size(); i++)
    System.out.println(copy.get(i));

我预计副本会,但副本的迭代会产生跟随错误

Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 1
    at org.objectweb.asm.tree.InsnList.toArray(InsnList.java:199)
    at org.objectweb.asm.tree.InsnList.get(InsnList.java:110)

标签: javajava-bytecode-asmjvm-bytecode

解决方案


虽然我无法重现您的特定异常,但这似乎只是您的方法存在根本问题的另一个症状。作为国家的文件AbstractInsnNode

表示字节码指令的节点。一条指令一次最多只能出现InsnList一次。

所以你不能复制一个InsnList,至少不能通过将相同的节点插入另一个列表。这也是从源列表中删除节点的add(InsnList)insert(InsnList)方法的原因。InsnList

当您真正想要 的副本时InsnList,您还必须创建节点对象的副本,这通过访问者 API 最容易完成:

private static InsnList copy(InsnList insnList) {
    MethodNode mv = new MethodNode();
    insnList.accept(mv);
    return mv.instructions;
}

但只要任务允许,您应该使用 ASM 的访问者 API 将指令直接写入目标类文件,而不是创建指令列表的副本。另请参阅有关您的实际任务的答案。


推荐阅读