你真的懂 i++ 和 ++i 吗?
对于 ++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
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构