public class Test {
static void print(int i){
System.out.println(i);
}
public static void main(String[] args) {
int i = 0;
print(i++);
print(++i);
}
}
通过一段简单的java代码实现了打印i++与++i的功能,其中i++打印出来的是0,而++i打印出来的是2。
0
2
通过使用jclasslib工具我们可以获取main方法的字节码
0 iconst_0
1 istore_1
2 iload_1
3 iinc 1 by 1
6 invokestatic #4 <Test.print>
9 iinc 1 by 1
12 iload_1
13 invokestatic #4 <Test.print>
16 return
jvm在运行时将class文件加载到方法区中,调用的相应的方法是就在方法区中寻找并加载到虚拟机栈中对应线程的栈空间,并为它分配一个栈帧,栈帧中包含局部变量表 , 操作数栈 ,动态连接 ,方法返回地址 和 一些额外的附加信息。
iconst_0将int 0压入栈中
istore_1将栈0位置的数存储在局部变量1位置中(虚拟机中局部变量0分配给了this,存储的位置都是从1开始,在静态方法中,没有this引用,0位置分配给形参args了)
iload_1将局部变量1中的数值压入栈中
iinc 1 by 1是将局部变量1中的数自增1
invokestatic #4 <Test.print>是调用print方法,同时将栈中0位置的数当作参数传入。
以上是print(i++)的字节码解析,我们可以看到虚拟机先将数值从局部变量中导入到栈中,然后在将局部变量的数值进行自增,所以导入print的参数依然是自增之前的数也就是0
同理,我们可以看到print(++i)的运行过程是先iinc 1 by 1,然后iload_1,先自增然后再把他推入栈中,所以print()传入的参数是自增之后的数值也就是2。