作者:年轻人创事业的美丽家园 | 来源:互联网 | 2023-10-10 20:52
我想大部分都知道i++和++i的区别,i++就是先拿i来使用,之后再自增加1,而++i则是先自增加1,在拿i来使用,例如对于下面这两个语句,我敢保证大部分人都会做: inti1;S
我想大部分都知道 i++ 和 ++i的区别,i++ 就是先拿i来使用,之后再自增加1,而++i则是先自增加1,在拿i来使用,例如对于下面这两个语句,我敢保证大部分人都会做:
int i = 1;
System.out.println(i++)
int i = 1;
System.out.println(++1)
答案分别为 1,2。对于这个答案我猜大多数人都能答出来。不过 i++ 和 ++i 这两个操作,在内部是如何实现的呢?
我们先来看另外一个问题:
public static void main(String[] args) {
int i = 1;
System.out.println(i+++i++);
System.out.println(i);
}
这个比刚才那个难了点,答案分别是3,3。假如你对这个答案的由来了如指掌,那么你大不可必往下看,假如你不大理解或者想从底层的汇编指令的来了解这个操作,那么你可以看看我的解释。
首先我们先来看看 i++ 的题,主要是为了后面好解释点。
int i = 1;
System.out.println(i++);
这两行代码的部分汇编指令如下,注意,我只列出了几个重点的汇编语句:
ICONST_1 //把常量 1 加载到栈顶
ISTORE 1 //把栈顶的元素弹出,并赋值给局部变量表中位置为“1”的变量,此时指变量i。这两句就相当于 int i = 1;
//接下来执行第二行代码
ILOAD 1 //把局部变量表中位置为“1”的变量加载到栈顶,即把i的值加载到栈顶
IINC 1 1 //直接把局部变量表中位置为“1”的变量加1,即把 i 加1。注意,这条指令并没有修改操作数栈就把 i 加1了。
INVOKEVIRTUAL java/io/PrintStream.println (I)V //把栈顶的元素打印出来,此时栈顶的元素是 1。所以打印的是 1
注:可以左右拉动
所以,此时打印的是1。
有些人可能没弄过汇编会有点蒙蔽,没事,我花个时间画个图来模拟(注:省略很多细节)。
刚开始时的局部变量表和操作数栈如图所示: