先看一段代码:
int i = 0;
int a = i++;
sout("a的值是:"+i);
sout("i的值是:"+i);
最终的编译之后的核心字节码如下
L0
BITPUSH 0//将常量0压入操作栈
ISTORE_1//将当前栈顶元素,弹出并保存到局部变量表的slot_1中L1
ILOAD_1//从局部变量表的第一个slot槽中,取出该值,压入操作栈顶
IINC 0,1//直接将slot槽中的值自增(+1)操作,注意此时是与当前栈无关的
ISTORE_2//将当前栈顶元素,弹出并保存到局部变量表的slot_2中L2
ILOAD_2//从局部变量表的第二个slot槽中,取出该值,压入操作栈顶
IRETURN//返回栈顶元素
这里有两个注意点:
IINC
的自增操作,并未影响当前的栈顶元素,并且 slot_1 中的元素自增完成后,已经由0变成了1ISTORE_2
弹出的栈顶元素值依旧是0,并未改变
a的值是: 0
i的值是: 1;
我这里画了一个图来帮助大家理解
文章图片
文章图片
文章图片
文章图片
文章图片
文章图片
再来看看++i
int i = 0;
int a = ++i;
sout("a的值是:"+i);
sout("i的值是:"+i);
对于++i 来说,对应的字节码如下,先自增再入栈,那么结果就很清晰了
最终的核心编译之后的字节码如下
L0
BITPUSH 0//将常量0压入操作栈
ISTORE_1//将当前栈顶元素,弹出并保存到局部变量表的slot_1中L1
IINC 0,1//直接将slot槽中的值自增(+1)操作
ILOAD_1//从局部变量表的第一个slot槽中,取出该值(该值此时已经自增过了),压入操作栈顶
ISTORE_2//将当前栈顶元素,弹出并保存到局部变量表的slot_2中L3
ILOAD_2//从局部变量表的第二个slot槽中,取出该值,压入操作栈顶
IRETURN//返回栈顶元素
最终的输出的结果为:
a的值是: 1
i的值是: 1;
总结
i++
和 ++i 在理论上的区别是 :i++
:是先把i拿出来使用,然后再+1;++i
:是先把i+1,然后再拿出来使用;
近期热文推荐:
1.1,000+ 道 Java面试题及答案整理(2022最新版)
【图解 i++ 和 ++i 的区别,看了必懂...】2.劲爆!Java 协程要来了。。。
3.Spring Boot 2.x 教程,太全了!
4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!
5.《Java开发手册(嵩山版)》最新发布,速速下载!
觉得不错,别忘了随手点赞+转发哦!
推荐阅读
- webgl|一文聊透 JVM 内存分布、内存对齐、压缩指针!
- Java|第1节 MySQL 架构篇 2021-12-24
- activiti|springboot集成activiti5.6 ,接口报401、403
- 面试技巧|2021 java面试题目(持续更新...)
- 编程|【整理】IDEA优化措施
- java|分享Java代码的一些小建议,脱离小白——学会优化代码50个方案
- 微服务|RabbitMQ之消息可靠性、死信交换机、惰性队列及集群
- java|iOS 高刷屏监控 + 优化(从理论到实践全面解析)
- 大数据|一次关于架构的“嘴炮”