【面试题】从JVM的角度去理解i++和++i

本文通过实例代码和JVM字节码分析,探讨Java中i++和++i操作的执行原理,揭示它们在JVM层面如何影响变量值的变化,帮助理解面试中常遇到的这个问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

public class Main {
    public static void main(String[] args) {
	    int i = 1;
	    i = i++;
	    int j = i++;
	    int k = i + ++i*i++;
        System.out.println("i="+i);
        System.out.println("j="+j);
        System.out.println("k="+k);
    }
}

实例代码如上,可以很容易得出答案是i=4,j=1,k=11。

要深究其原理,首先要反编译它的字节码文件,这里我就截取我们要进行论述的部分:

L0/1/2/3分别代表的分别是main方法当中的第1/2/3/4行代码
LINENUMBER x这里的x表示数字,代表这一段操作码对应的是原代码当中的第x行
ICONST_1将int类型的值1入操作数栈
ISTORE 1将操作数栈顶的值保存到局部变量1当中
ILOAD 1将局部变量1的值装载成int类型入操作数栈
IINC 1 1将后一位数的值加到前一位数的局部变量当中去(这里就是将1加到局部变量1当中去)
IMUL将操作数栈顶的两个int类型数相乘,然后结果入栈
IADD将操作数栈顶的两个int类型数相加,然后结果入栈

 (1)拿main方法内部第1行代码:int i = 1;来说,它对应的是L0块里的指令操作码:

   L0
    LINENUMBER 5 L0 //代表原代码当中第5行代码
    ICONST_1  //将int类型值为1的数入操作数栈
    ISTORE 1 //将操作数栈顶的值保存到变量1当中

对应下图所示过程:

 

(2)i = i++;

   L1
    LINENUMBER 6 L1
    ILOAD 1  //将局部变量表当中的第一个变量(i)的值入操作数栈
    IINC 1 1 //将第一个局部变量也就是i=1加上一个int类型的1,也就变成了i=2
    ISTORE 1 //将操作数栈顶值保存为第一个变量(i)

 这里i的值曾经变为2,但是后来被覆盖了。

(3)int j = i++;

   L2
    LINENUMBER 7 L2
    ILOAD 1 //将局部变量1入操作数栈
    IINC 1 1 //将局部变量1的值加1
    ISTORE 2 //将操作数栈栈顶当前的值1存储为局部变量2

 

 (4)int k = i + ++i*i++;

   L3
    LINENUMBER 8 L3
    ILOAD 1 //将局部变量1入操作数栈
    IINC 1 1 //将局部变量1的值加1 (此时i=3)
    ILOAD 1 //将局部变量1入操作数栈
    ILOAD 1 //将局部变量1入操作数栈
    IINC 1 1 //将局部变量1的值加1 (此时i=4)
    IMUL //将栈顶两个int类型数相乘,结果入栈
    IADD //将栈顶两个int类型数相加,结果入栈
    ISTORE 3 //将当前栈顶值保存为局部变量3,也就是k

 

(5)综上分析,结果和预期相同。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值