最近看到个面试题:
public static void f4() {
int j=0;
for(int i=0;i<100;i++){
j = j++;
}
System.out.println(j);
}
输出结果是0,如果换成j++,那么输出100,这是为什么?
对于这种问题,其实有点经验的程序员都知道,前置++和后置++的运算区别:
1、++j是先将变量的值加1,然后使用加1后的值参与运算
2、j++是先使用该值参与运算,然后再将该值加1(操作的是栈中的数据!!!)
话虽这么说,那怎么来证明上面两点呢?
万物皆有始有终,对于代码,只能从字节码的角度看看是不是能挖掘有价值的信息。
在讲字节码之前,先简单的了解下Java栈,在JVM中有这么一个数据结构叫Java栈,当线程启动的时候,会分配一块内存当做该线程的栈,每个栈由一系列的栈帧组成。每个栈帧对应一个方法,当线程执行方法时,就是栈帧出栈入栈的过程。
每个栈帧包含三部分数据:本地变量(参数+方法内的变量)、操作数栈和其它数据,本文主要涉及本地变量和操作数栈。
先看看后置++的实现:
public static void main(String[] args){
int i= 0;
i = i++;
}
编译之后的字节码 javap -v Test.class > test.txt 如下:
iconst_0 //把数值0 push到操作数栈
istore_1 // 把操作数栈写回到本地变量第2个位置
iload_1 // 把本地变量第2个位置的值push到操作数栈
iinc 1,1 //把本地变量表第2个位置加1 -----------在赋值给i之前 本地变量表+1!!!!
istore_1 // 把操作数据栈写回本地变量第2个位置
整个过程这样的:
可以发现变量a在执行iinc 1,1的时候已经变成1了,但是istore_1又把变量a所在位置覆盖成了0,所以执行完i = i++,i还是原来那个值。
另外,来看下前置++的实现:
字节码就不再解释了,整个过程实现如下:
和后置++不同的地方在于,在变量进入操作数栈之前,就先执行了iinc指令,所以进入操作数的值是加1后的值,最后写回的值也是最新值。