ASM 创建带有参数名称的方法 w/ ClassWriter

ASM create method with parameter name w/ ClassWriter

使用 org.objectweb.asm 库的 Kotlin 代码

import org.objectweb.asm.ClassWriter
import org.objectweb.asm.Opcodes.*

fun main(args: Array<String>) {
    val classWriter = ClassWriter(ClassWriter.COMPUTE_MAXS)

    classWriter.visit(
            V1_8,
            ACC_PUBLIC,
            "com/github/patrick/learnasm/build/HelloWorld",
            null,
            "java/lang/Object",
            null
    )

    val methodVisitor = classWriter.visitMethod(
            ACC_PUBLIC + ACC_STATIC,
            "main",
            "([Ljava/lang/String;)V",
            null,
            null
    )

    methodVisitor.visitFieldInsn(
            GETSTATIC,
            "java/lang/System",
            "out",
            "Ljava/io/PrintStream;"
    )

    methodVisitor.visitLdcInsn(
            "Hello World!"
    )

    methodVisitor.visitMethodInsn(
            INVOKEVIRTUAL,
            "java/io/PrintStream",
            "println",
            "(Ljava/lang/String;)V",
            false
    )

    methodVisitor.visitInsn(
            RETURN
    )

    methodVisitor.visitMaxs(
            0,
            0
    )

    methodVisitor.visitEnd()

    val bytes = classWriter.toByteArray()
    val clazz = defineClass(
            "com.github.patrick.learnasm.build.HelloWorld",
            bytes,
            0,
            bytes.count()
    )

    clazz.getDeclaredMethod("main", Array<String>::class.java).invoke(null, null)
}

我正在尝试学习 asm,我很好奇是否可以创建一个带有命名参数的方法。

例如在这种情况下,创建的 class 没有参数名称(在反编译器中默认为 var0),我希望我可以保存参数名称,如“args”。有可能吗?

通过查看已编译的 class(包括调试信息)找到答案

简单的方法就是模拟编译后的class

我所要做的就是在上面贴上一些标签,然后访问带有名称的局部变量。

            methodVisitor = classWriter.visitMethod(
                    Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC,
                    "main",
                    "([Ljava/lang/String;)V",
                    null,
                    null
            )

            methodVisitor.visitParameterAnnotation(
                    0,
                    "Lorg.jetbrains.annotations.NotNull;",
                    false
            )

            val start = Label()
            methodVisitor.visitLabel(start)

            methodVisitor.visitFieldInsn(
                    Opcodes.GETSTATIC,
                    "java/lang/System",
                    "out",
                    "Ljava/io/PrintStream;"
            )

            methodVisitor.visitLdcInsn("Hello World!")

            methodVisitor.visitMethodInsn(
                    Opcodes.INVOKEVIRTUAL,
                    "java/io/PrintStream",
                    "println",
                    "(Ljava/lang/String;)V",
                    false
            )

            methodVisitor.visitInsn(Opcodes.RETURN)

            val end = Label()
            methodVisitor.visitLabel(end)

            methodVisitor.visitLocalVariable(
                    "args",
                    "[Ljava/lang/String;",
                    null,
                    start,
                    end,
                    0
            )

            methodVisitor.visitMaxs(0, 0)
            methodVisitor.visitEnd()