美文网首页
用asmtools.jar 修改和生成class文件

用asmtools.jar 修改和生成class文件

作者: 你弄啥来 | 来源:发表于2019-10-11 16:44 被阅读0次

    本次实验需要的asmtools.jar的文件 需要给个链接:https://github.com/hengyunabc/hengyunabc.github.io/files/2188258/asmtools-7.0.zip

    最近在看一篇关于 郑雨迪 讲的 Java代码是怎么运行的文章时,文章最后留了个小作业。

    echo '
    public class App{
     public static void main(String[] args) {
      boolean flag = true;
      if (flag) System.out.println("Hello, Java!");
      if (flag == true) System.out.println("Hello, JVM!");
     }
    }' > App.java
    $ javac App.java
    $ java App
    $ java -cp /path/to/asmtools.jar org.openjdk.asmtools.jdis.Main Foo.class > Foo.jasm.1 
    $ awk 'NR==1,/iconst_1/{sub(/iconst_1/, "iconst_2")} 1' Foo.jasm.1 > App.jasm
    $ java -cp /path/to/asmtools.jar org.openjdk.asmtools.jasm.Main App.jasm
    $ java App
    

    完全找不到头绪觉得flag恒为true没什么好研究的,后来了解了asmtools.jar 和 awk 是做什么的才知道老师的用意,并get到了一个新技能--------------直接去修改一个在不修改java代码的情况下如何去修改class文件。

    //改行代码是将Foo.class 文件通过asmtools.jar转换成 Foo.jasm.1  
    java -cp /path/to/asmtools.jar org.openjdk.asmtools.jdis.Main Foo.class > Foo.jasm.1 
    //awk 命令是linux中特有的一种文本处理器 就是在通过正则将iconst_1 换成 iconst_2
     awk 'NR==1,/iconst_1/{sub(/iconst_1/, "iconst_2")} 1' Foo.jasm.1 > Foo.jasm
    //再把 jasm文件转换成class 文件 再执行 java App的时候发现只有打印出 Hello Java了 
     java -cp /path/to/asmtools.jar org.openjdk.asmtools.jasm.Main App.jasm
    

    通过 java -cp /path/to/asmtools.jar org.openjdk.asmtools.jdis.Main Foo.class > Foo.jasm.1 获得到的Foo.jasm.1的文件 看上去有点像字节码文件

    [root@iZ2vc5vqvvbnxyz8xrvnvpZ asmtooltest]# cat App.jasm.1
    
    super public class App
            version 52:0
    {
    
    
    public Method "<init>":"()V"
            stack 1 locals 1
    {
                    aload_0;
                    invokespecial   Method java/lang/Object."<init>":"()V";
                    return;
    }
    public static Method main:"([Ljava/lang/String;)V"
            stack 2 locals 2
    {
                    iconst_1;
                    istore_1;
                    iload_1;
                    ifeq    L14;
                    getstatic       Field java/lang/System.out:"Ljava/io/PrintStream;";
                    ldc     String "Hello Java!";
                    invokevirtual   Method java/io/PrintStream.println:"(Ljava/lang/String;)V";
            L14:    stack_frame_type append;
                    locals_map int;
                    iload_1;
                    iconst_1;
                    if_icmpne       L27;
                    getstatic       Field java/lang/System.out:"Ljava/io/PrintStream;";
                    ldc     String "Hello,JVM";
                    invokevirtual   Method java/io/PrintStream.println:"(Ljava/lang/String;)V";
            L27:    stack_frame_type same;
                    return;
    }
    } // end Class App
    

    执行完 awk 'NR==1,/iconst_1/{sub(/iconst_1/, "iconst_2")} 1' App.jasm.1 > App.jasm 命令后 会新生成一个App.jasm的文件 发现 第一个 iconst_1 变成了 iconst_2

    cat App.jasm
    
    super public class App
            version 52:0
    {
    
    
    public Method "<init>":"()V"
            stack 1 locals 1
    {
                    aload_0;
                    invokespecial   Method java/lang/Object."<init>":"()V";
                    return;
    }
    
    public static Method main:"([Ljava/lang/String;)V"
            stack 2 locals 2
    {
                    iconst_2;
                    istore_1;
                    iload_1;
                    ifeq    L14;
                    getstatic       Field java/lang/System.out:"Ljava/io/PrintStream;";
                    ldc     String "Hello Java!";
                    invokevirtual   Method java/io/PrintStream.println:"(Ljava/lang/String;)V";
            L14:    stack_frame_type append;
                    locals_map int;
                    iload_1;
                    iconst_1;
                    if_icmpne       L27;
                    getstatic       Field java/lang/System.out:"Ljava/io/PrintStream;";
                    ldc     String "Hello,JVM";
                    invokevirtual   Method java/io/PrintStream.println:"(Ljava/lang/String;)V";
            L27:    stack_frame_type same;
                    return;
    }
    } // end Class App
    

    执行完 java -cp /path/to/asmtools.jar org.openjdk.asmtools.jasm.Main App.jasm 命令时 会生成一个新的App.class文件,在执行的时候就能看到效果了。

    相关文章

      网友评论

          本文标题:用asmtools.jar 修改和生成class文件

          本文链接:https://www.haomeiwen.com/subject/yofxmctx.html