对于 ++i 和 i++,许多人可能都知道,不就是先加1再取值,和先取值再加1嘛。然而,真的是这样吗?请先看以下4道题,能全部答对可以忽略这篇文章。
题目
// 示例1
int i = 1;
i = i++;
System.out.println("i = " + i);
// 示例2
int i = 1;
int j = (2 * i++) + i;
System.out.println("j = " + j);
// 示例3
int i = 1;
int j = i + (2 * i++);
System.out.println("j = " + j);
// 示例4
int i = 1;
int j = 1;
int k = i++ + ++i + ++j + j++;
System.out.println("k = " + k);
先别着急看答案,先自己思考下,解出自己的答案,然后再往下翻查看答案是否与你的一致。
答案
示例1:i = 1
示例2:j = 4
示例3:j = 3
示例4:k = 8
你是否发现有些答案和你想的不一样,如果我告诉你 ++i 和 i++ 其实都是先计算加1,你是不是更懵逼了!在详解答案之前,先了解两个知识点。
1 i++ 和 ++i 原理
i++
:先自增,再返回自增之前的值++i
:先自增,再返回自增之后的值- 不论是前++还是后++,它们的共同点就是
先自增
。
2 表达式原则
一个变量也是表达式,多个表达式的加减法运算都是从左到右进行的,当然乘除法的优先级还是大于加减法的。
答案详解
// 示例1 结果:i = 1
int i = 1;
i = i++;
System.out.println("i = " + i);
根据原理,先自增,再返回自增之前的值
,i 自增后,i = 2,然后返回自增之前的值1,此时表达式变成 i = 1,1没赋值给 i 时 i 的值是2,但最后把1赋值给 i 时,i 的值就又变成1了。
// 示例2 结果:j = 4
int i = 1;
int j = (2 * i++) + i;
System.out.println("j = " + j);
根据表达式原则,一个变量也是表达式,多个表达式的加减法运算都是从左到右进行的
- 优先运算左边的表达式,即(2 * i++),i++后,i 的值为2,并返回自增之前的值1
- 此时表达式为 int j = (2 * 1) + i ,i 的值已经是2了
- 最后表达式变为 int j = (2 * 1) + 2 ,于是 j = 4。
// 示例3 结果:j = 3
int i = 1;
int j = i + (2 * i++);
System.out.println("j = " + j);
按数学思维,我们可能会先计算 2 i++ 部分,i 先自增 i = 2,然后返回自增之前的值1,此时表达式变为 int j = i + (2 1) 。此时 i 的值为2了,故表达式又变为 int j = 2 + (2 * 1) ,结果 j = 4,然而这答案是错误的。正确逻辑如下:
根据表达式原则,一个变量也是表达式,多个表达式的加减法运算都是从左到右进行的
。
- int j = i + (2 * i++) 先算 + 号左边 i 这个表达式,表达式的结果为1
- 表达式变为 j = 1 + (2 * i++)
- 再计算 + 号右边的表达式,然后 i 自增并返回自增之前的值1,表达式又变为 j = 1 + (2 * 1)
- 最终结果为 j = 3,此时 i 的值为2
// 示例4 结果:k = 8
int i = 1;
int j = 1;
int k = i++ + ++i + ++j + j++;
System.out.println("k = " + k);
- 先计算 i++,i++ 之后 i 的值为2,并返回自增之前的值1,表达式变为 1 + ++i + ++j + j++。此时的 i 值为2
- 再计算 ++i,++i 之后 i 的值为3,并返回自增之后的值3,表达式变为 1 + 3 + ++j + j++。此时 i 的值为3
- 再计算 ++j,++j 之后 j 的值为2,并返回自增之后的值2,表达式变为 1 + 3 + 2 + j++。此时 j 的值为2
- 再计算 j++,j++之后 j 的值为3,并返回自增之前的值2,表达式变为 1 + 3 + 2 +2,即结果为8,此时j的值为3