public class Count {
public static void main(String[] args) {
int i = 0;
i = i++ ;
System.out.println(i);
}
}
上面代码输出的i为0,如果是把i = i++换成i=++i,又会输出1,这是由于i++是先赋值,再计算导致,但是为什么先赋值呢?
public static void main(String[] args) {
int i = 0;
i++ ;
}
public static void main(java.lang.String[]);
Code:
0: iconst_0 将int型0推送至栈顶(找到0)
1: istore_1 将栈顶int型数值存入第二个本地变量(变量就是i)
2: iinc 将指定int型变量增加指定值(如i++, i--, i+=2等)
5: return
上图是i++的字节码,++i的和i++的一摸一样,0和1就是最基础的给i赋值为0,iinc就是把i自增。
但是如果加上一个i=i++呢?
// i = i++;
public static void main(java.lang.String[]);
Code:
0: iconst_0 将int型0推送至栈顶
1: istore_1 将栈顶int型数值存入第二个本地变量
2: iload_1 将第二个int型本地变量推送至栈顶
3: iinc 将指定int型变量增加指定值(如i++, i--, i+=2等)
6: istore_1 将栈顶int型数值存入第二个本地变量
7: return
0和1依旧是给i赋值为0,2是把当前i的值推送到栈顶,然后iinc对局部变量表的i进行自增,6是把栈顶的数值赋给i,注意,2的时候推送到栈顶的是0,所以又变回0了。
// i = ++i
public static void main(java.lang.String[]);
Code:
0: iconst_0 将int型0推送至栈顶
1: istore_1 将栈顶int型数值存入第二个本地变量
2: iinc 将指定int型变量增加指定值(如i++, i--, i+=2等)
5: iload_1 将第二个int型本地变量推送至栈顶
6: istore_1 将栈顶int型数值存入第二个本地变量
7: return
i = ++i的时候,iinc往前提了一步,说明,在i的值被压到栈顶前,先进行了自增,所以栈顶的值是1,此时再赋值给局部变量表的里i,就是自增过的。
总结:iinc操作是直接针对局部变量表的,不经过操作栈,i++所谓的先赋值,从字节码层面来看,就是i在进行自增前,先把原来的值存到了操作数栈,所以如果执行b = i++,b之所以等于是i原来的值,就是由于操作数栈的值是自增前的值,而++i的,自增操作提前了,所以b=++i后,b的值和i的值一样。