一 概述
对于Java程序而言,我们可以利用javac命令对源码进行编译,使得源文件转换成含有字节码的class文件,而对于class文件我们既可以使用java命令通过JVM对class文件的加载解析并生成机器可识别的机器码指令。我们也可以通过javap命令对class文件进行反编译。
二 反编译命令手册
三 反编译过程
源代码
package com.interview.javabasic.bytecode;
public class ByteCodeSample {
public static void main(String[] args) {
int i=1,j=5;
i++;
++j;
System.out.println(i);
System.out.println(j);
}
}
javac编译后的字节码(不知所云)
javap实现反编译后的代码及分析
javap -c xxx.class //对编译后的class文件实现反编译
代码
Compiled from "ByteCodeSample.java"
public class com.interview.javabasic.bytecode.ByteCodeSample {
public com.interview.javabasic.bytecode.ByteCodeSample();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public static void main(java.lang.String[]);
Code:
0: iconst_1
1: istore_1
2: iconst_5
3: istore_2
4: iinc 1, 1
7: iinc 2, 1
10: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
13: iload_1
14: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
17: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
20: iload_2
21: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
24: return
}
代码分析:
Compiled from "ByteCodeSample.java" //说明是从哪个类编译而来
//声明一个带有package(com.interview.javabasic.bytecode.ByteCodeSample)的类
public class com.interview.javabasic.bytecode.ByteCodeSample {
//当我们没有指定构造函数的时候,编译器会默认生成一个无参构造函数
public com.interview.javabasic.bytecode.ByteCodeSample(); //无参构造函数
Code: //表示需要执行的内容
0: aload_0 //在非静态方法中表示对this对象的操作,在Static方法中,表示对方法的第一个参数的操作
1: invokespecial #1 //调用父类构造方法Super()去进行初始化 // Method java/lang/Object."<init>":()V
4: return //构造函数执行完成并退出
//main函数接收的是一个字符串数组:java.lang.String[]
public static void main(java.lang.String[]);
Code: //需要执行的内容,此处涉及到Java中的堆和栈,由于局部变量所以涉及到栈的操作
0: iconst_1 //将常量1放到栈顶
1: istore_1 //将栈顶的常量1放到局部变量1(即第一个局部变量)中
2: iconst_5 //将常量5放到栈顶
3: istore_2 //将栈顶的常量5放到局部变量2(即第二个局部变量)中
4: iinc 1, 1 //将变量1加上1
7: iinc 2, 1 //将变量2加上1
10: getstatic #2 //获取printStream的静态域,并将其压入栈顶 // Field java/lang/System.out:Ljava/io/PrintStream;
13: iload_1 //将本地第一个变量的值推送到栈顶
14: invokevirtual #3 //调用PrintStream的println方法打印第一个变量的值 // Method java/io/PrintStream.println:(I)V
17: getstatic #2 //获取printStream的静态域,并将其压入栈顶 // Field java/lang/System.out:Ljava/io/PrintStream;
20: iload_2 //将本地第二个变量的值推送到栈顶
21: invokevirtual #3 //调用PrintStream的println方法打印第二个变量的值 // Method java/io/PrintStream.println:(I)V
24: return //方法执行完毕并退出
}