一个表达式中对同一个变量多次赋值(i = i++)

问:

int i = 0;
i = ++i;
i的值是多少?毫无疑问,i = 1;但是,i = ++i;这条语句对i进行了两次赋值:第一次,i自增,此时i=1;第二次,将++i的表达式的值(也就是1)赋给i。我们通过字节码来分析一下这段代码的执行过程:

//int i = 0;的字节码
0:iconst_0 //将int型0推送至栈顶
1:istore_1 //将栈顶值(0)存入局部变量表中索引为1的地方(也就是i)

// i = ++i;的字节码
2:innc 1,1  //在局部变量表中索引为1的变量加1,此时i=1;
3:iload_1   //索引为1的变量值存入栈顶
4:istore_1  //将栈顶值存入索引为1的变量中

理解了上面的代码后,再考虑这样两行代码:

int i = 0;
i = i++;
这两条代码执行完成后,i的值是多少?答案是0。稍加思考,就会觉得这个结果并不是太意外,这两句代码的执行过程是这样的:

1))定义整型局部变量i,初始值为0;

2)JVM执行i++计算,该表达式的值为0;

3) i自增,i此时为1;

4)将i++表达式的值赋给i,i++表达式的值是0,所以赋值后,i = 0;

5)计算完毕,i = 0。

上面的每一步都不会让我们感到意外,为了更加清晰的说明问题,上面两行代码的等价代码如下:

int i = 0;

int tmp = 0;  //对应分析中的第(2)步,也就是i++表达式的值
i++;         //分析中的第(3)步
i = tmp;    //分析中的第(4)步

我们也从字节码看看这两条语句的执行过程:

//int i = 0;的字节码

0:iconst_0 //将int型0推送至栈顶,此时栈中的内容:[0]

1:istore_1 //将栈顶值(0)存入局部变量表中索引为1的地方(也就是i),i = 0;栈中的内容:[ ]

// i = i++;的字节码

2:iload_1 //索引为1的变量值存入栈顶,也就是i++表达式的值(0)存入栈顶,此时栈中的内容:[0]

3:innc 1,1 //在局部变量表中索引为1的变量加1,此时i=1;栈中的内容:[0]

4:istore_1 //将栈顶值(0)存入索引为1的变量中,i=0;栈中的内容:[ ]

从字节码中,我们可以看到:JVM首先把i++表达式的值存入到操作数栈中,也就是栈中保存着i = i++右半部的值(表达式i++,也就是0);然后把=右边的表达式计算完毕;最后将表达式i++的值赋给i(栈顶元素出栈,赋给i)。注意,i++是一个表达式,这个表达式的值是0。

这样的讨论其实是没有意义的,不要在一个表达式中,对同一个变量多次赋值!

参考资料:http://www.ticmy.com/?p=43(写这篇博客主要是为了加深自己的理解)


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 这段代码的执行过程如下: 1. 首先给 i 和 j 赋值为 5。 2. 执行赋值语句 p = (i++) + (i++) + (i++)。 3. i++ 表示先取 i 的值,再将 i 加 1。 4. 因此 i++ + i++ + i++ 相当于 5 + 6 + 7,即 i 的值分别为 5、6、7 时的和。 5. 将结果赋值给 p,即 p = 5 + 6 + 7 = 18。 6. 最终 i 的值为 8,因为 i++ 执行了三次。 所以,实际上根据代码的执行结果,p 应该等于 18,而不是 15。 ### 回答2: 在C语言,当一个表达式有多个操作符时,按照运算符优先级的顺序执行,乘法运算符具有比加法运算符更高的优先级,因此在表达式`p = (i ) (i ) (i )`,先执行两个乘法运算。 首先,根据赋值运算符的左结合性,首先计算`(i * i)`的值。`i`的值为5,那么`(i * i)`的值为25。 然后,由于没有指定运算符的优先级关系,C语言会按照从左到右的顺序依次执行剩下的乘法运算和加法运算。这就意味着,`(i * i) * i`的运算会在`(i * i)`的运算后执行。`(i * i * i)`的值为125。 最后,将得到的结果125赋值变量`p`,所以`p`的值为125。 至于为什么你说等于15,可能是因为在表达式存在公式的缺失或拼写错误。 ### 回答3: 这段代码,首先定义了整型变量i和j,并且给它们赋予初始值都为5。然后定义了两个未初始化的整型变量p和q。 接下来,通过表达式p = (i ) (i ) (i )对p进行赋值。在C语言,这里的表达式用到了运算符的结合性和优先级,即从左到右依次计算。根据运算符优先级规则,首先会计算括号的算术运算表达式。 这里括号表达式(i ) (i ) (i )实际上是一个自增运算的连续执行。它等同于三个独立的自增运算i++,因为这里的括号是多余的,可以忽略。 而自增运算符++的功能是将变量的值加1,并返回加1之后的值。所以,i在第一个自增运算i++之后的值变成了6,在第二个自增运算i++之后的值变为了7,在第三个自增运算i++之后的值变成了8。 最后,将这个连续的自增运算的返回值累加起来,即6+7+8=21,并将结果赋给变量p。所以最终,p的值为21。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值