java - 如何使用 ASM 创建局部变量?

标签 java variables local bytecode java-bytecode-asm

我正在尝试使用 ASM 修补一个类。我需要在函数中添加一些逻辑。此逻辑需要一个新的局部变量。这是我所做的:

class CreateHashTableMethodAdapter extends MethodAdapter {
    @Override  
    public void visitMethodInsn(int opcode, String owner,String name, String desc){
        System.out.println(opcode + "/" + owner + "/" + name + "/" + desc);

        if(opcode == Opcodes.INVOKESPECIAL &&
                "javax/naming/InitialContext".equals(owner) &&
                "<init>".equals(name) &&
                "()V".equals(desc)){

            System.out.println("In mod");
            //  83: new #436; //class javax/naming/InitialContext
            //  86: dup


            mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "javax/naming/InitialContext", "<init>", "()V");
            mv.visitVarInsn(Opcodes.ASTORE, 1);

            Label start_patch = new Label();
            Label end_patch = new Label();

            mv.visitLabel(start_patch);



            mv.visitTypeInsn(Opcodes.NEW,"java/util/Hashtable");
            mv.visitInsn(Opcodes.DUP);
            mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/util/Hashtable", "<init>", "()V");  
            mv.visitVarInsn(Opcodes.ASTORE,9);

            // ........ sNip ..........

            mv.visitLabel(end_patch);
            mv.visitLocalVariable("env","Ljava/util/Hashtable;",null,start_patch,end_patch,9);

            //    127:  astore_1

        }
        else {
            mv.visitMethodInsn(opcode, owner, name, desc);
        }
    }
}

当我针对 CheckClassAdapter 运行此方法适配器时,它指出:

org.objectweb.asm.tree.analysis.AnalyzerException: Error at instruction 51: Trying to access an inexistant local variable 9
.... sNiP ....
00050 R R . . .  : R R  :     INVOKESPECIAL java/util/Hashtable.<init> ()V
00051 R R . . .  : R  :     ASTORE 9

我想我误用了 visitLocalVariable,但我找不到应该在哪里调用它。

当我 javap 生成字节码(不检查)时,我得到以下局部变量表:

LocalVariableTable: 
Start  Length  Slot  Name   Signature
91      40      9    env       Ljava/util/Hashtable;
0      343      0    this       Lpmu/jms/ServerJMS;
132      146      1    initialContext       Ljavax/naming/InitialContext;
153      125      2    topicConnectionFactory       Ljavax/jms/TopicConnectionFactory;
223      55      3    topic       Ljavax/jms/Topic;
249      29      4    topicSubscriber       Ljavax/jms/TopicSubscriber;
279      55      1    ex       Ljava/lang/Exception;
281      53      2    codeMessage       I
289      45      3    params       Lpmu/data/Parameters;
325      9      4    messageError       Ljava/lang/String;

您可能会注意到,我的变量在这里但最顶层?! 有什么想法吗?

最佳答案

创建新局部变量的一种便捷方法是扩展 LocalVariablesSorter 而不是 MethodAdapter。然后,您可以根据需要使用 newLocal() 分配局部变量,而不会干扰现有变量。请参阅 ASM homepage 上的 ASM 4.0 A Java 字节码工程库 的第 3.3.3 节了解详情。

关于java - 如何使用 ASM 创建局部变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13589924/

相关文章:

java - 不使用时区 API 从用户当前地理位置获取时区

php - Laravel Eloquent 类变量文档

sql-server - 使用 SQL Server Express 版本而不是 Compact 版本作为本地数据库缓存?

python - 如何添加功能 "see other computer on this local network"?

facebook - 本地货币支付/重大更改不起作用/错误代码 : 1353028

java - 使用 Java 代码未完全下载文件

Java - 在本地端口上监听 RTP 数据包

java - Rad 上传 Java 小程序和 z-index

javascript - 不能在括号中使用变量只能直接输入将dms转换为dd

javascript - 如何在 HTML 元素中插入两个 javascript 变量?