"i=i++"在java等高级语言的解释
前言
最近学习我在的团队招人,面试。我出了一道c的题。
int main(){
int i = ;
i+=++i;
printf("%d",i);
}
别喷!今晚学长们说,java,c#中完全不同,我就千方百计试试。
i++ 的解释
例子:
public class Test { public static void main(String[] args) { int i = 10; i = i++; System.out.println(i); } }
i = i++;则输出得到i=10;
而如果同样的代码换成C/C++执行,则无论是i=i++;得到的结果是相同的,都是11,这是为什么呢?
其实就 i = i++;这个语句而言,其实很无聊,谁又会写这样一个语句呢(写错除外^_^),因为当你声明了 i = 0;后,要么你使用0,即直接使用i,要么你使用1,即用i++或者++i;没有必要写i=i++;所有这样的语句很无聊。但是正是这么个无聊的语句,引发了一个关于编译器编译机制的大问题的讨论!为什么java得到i=10;而C/C++得到的是11呢
这是由于Java和C/C++采用的编译器不同,编译器不同,所以编译机制也不同,大家可以通过javap命令反编译出java虚拟机指令例如如下一段程序:
public class Test { public static void main(String[] args) { int i = 0; i = i++; System.out.println(i); } }
通过运行命令:javap –c Test 得到如下结果:
对于上图中显示的结果大家可能读不懂,那么我在这里解释一下,从Code 0开始:0: iconst_0 //将int类型常量0压入栈(栈顶)
这里大家可能会犯嘀咕,int i=0;i不是变量么?
i 的确是变量,但赋值号后面的那个0是常量,可以这么说,对于基本数据类型的值(注意是值),都是常量。这区别于引用类型,比如Student s = new Student(); 赋值号后面的是一个对象,它是放在堆中的,而前面的这个s仍然是变量,是放在栈中的。
如图:
解释:
1:istore_1 //将int类型值存入局部变量1
2: iload_1//从局部变量1中装载int类型值。也就是说,这个时候把i变量中的值拿出来了,压入栈(栈顶),因为这里执行的是i++, 也就是先使用后++,所有这句的目的是想把i在++之前的值先找个地方保存起来。
3:iinc 1,1// 把常量值1加到局部变量1上。这句话执行后i变量中存放的值就不是0了,而是1,注意:刚才压入栈的那个值,也就是i在++之前的值,即栈顶的值仍然是0
6:istore_1//把栈顶的int类型的值存入变量1中。这句话就是用栈顶的0把i变量中的1给替换掉了。
所以最后得出了i=0的结果。
用几句通俗易懂讲一遍:
Java在执行 变量=i++; 在执行变量赋值语句(不是常量赋值,而是例如:i = a;即赋值号后面有变量)之前,会先把i的值找个地方保存起来,然后直接在i变量的空间内对值+1,然后再把刚才保存起来的那个值赋给变量。
那么清楚了上面的过程后,我们再来看看j=i++;的JVM指令:
过程是这样的:
0:把int类型的常量0压入栈顶
1:把栈顶int类型的值存入变量1中(即给i赋值为0)
2:把int类型的常量0压入栈顶
3:把栈顶int类型的值存入变量2中(即给j赋值为0)
4:把变量1中的int类型的值装载到栈
5: 把数值1加到变量1中(给i变量的值加1)
8:把栈顶int类型的值存入变量2(即把0赋给j)
所以,这时i的值为1,j的值为0
小结
所以大家在使用java中的i++时一定要注意,其他的没什么问题,只有i=i++;有问题,这和C/C++截然不同。
"i=i++"在java等高级语言的解释的更多相关文章
- Java路径操作具体解释
1.基本概念的理解 绝对路径:绝对路径就是你的主页上的文件或文件夹在硬盘上真正的路径.(URL和物理路径)比如: C:\xyz\test.txt 代表了test.txt文件的绝对路径.http://w ...
- Java String类具体解释
Java String类具体解释 Java字符串类(java.lang.String)是Java中使用最多的类,也是最为特殊的一个类,非常多时候,我们对它既熟悉又陌生. 类结构: public fin ...
- 设计模式 - 迭代模式(iterator pattern) Java 迭代器(Iterator) 详细解释
迭代模式(iterator pattern) Java 迭代器(Iterator) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy 參考迭代器模式(ite ...
- “全栈2019”Java第九章:解释第一个程序
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- Java 序列化Serializable具体解释(附具体样例)
Java 序列化Serializable具体解释(附具体样例) 1.什么是序列化和反序列化 Serialization(序列化)是一种将对象以一连串的字节描写叙述的过程:反序列化deserializa ...
- 另外几种Java集合框架具体解释续
另外几种Java集合框架具体解释续 作者:chszs,未经博主同意不得转载.经许可的转载需注明作者和博客主页:http://blog.csdn.net/chszs fastutil库优于Trove库的 ...
- 【Java面试题】解释内存中的栈(stack)、堆(heap)和静态存储区的用法
Java面试题:解释内存中的栈(stack).堆(heap)和静态存储区的用法 堆区: 专门用来保存对象的实例(new 创建的对象和数组),实际上也只是保存对象实例的属性值,属性的类型和对象本身的类型 ...
- Java中“指针”的解释以及对“引用”的理解
Java中"指针"的解释以及对"引用"的理解 初学Java面对对象编程,对于一些概念还真的有点难以理解,主要是因为不由自主的联系到以前学过的C语言知识,时不时的 ...
- JAVA与编译语言及解释语言的关系
转自JAVA结合了编译和解释执行的优点 编译型语言是一次性编译成机器码,脱离开发环境独立运行,所以运行效率较高,但是由于编译成的是特定平台上机器码,所以可移植性差. 编译型语言的典型代表有C.C++. ...
随机推荐
- 结果集ResultSet
我们访问数据库时候经常见到这样遍历结果集 conn = DBHelper.getConnection(); String sql = "select * from items"; ...
- maven 不能使用 snapshot 的解决方式
最近项目需要用到snapshot的包来进行构建过程,但是怎么都下不了构建的snapshot包.查询了相关资料,发现网上的资料不全,特总结下: 我使用的是nexus来作为代理中央库proxy. 检查步骤 ...
- python 字典中的get()方法
https://blog.csdn.net/weixin_38705903/article/details/79231551
- REdis CPU百分百问题分析
REdis版本:4.0.9 运行环境:Linux 3.10.107 x86_64 gcc_version:4.8.5 结论:是一个BUG,在4.0.11版本中被作者antirez所修复 现象: 1) ...
- submit插件安装的问题与集成了插件的submit
写在最前面,方法有二种.一种是在线安装,这种办法我尝试过,受网速和软件卡顿的问题,进行不顺利.第二种就是我下面介绍的这种,手动安装. 最精华的在后面,可以直接跳转到最后.我找了很久的,最新版的汉化,而 ...
- java 判断用户是PC端和还是APP端登陆
java 判断用户是PC端和还是APP端登陆 public void getRequestHeader(HttpServletRequest request){ // 从浏览器获取请求头信息 Stri ...
- 团队作业第六周--alpha阶段项目复审
组名 优点 缺点 排名 天冷记得穿秋裤队 支持文件离线下载,没有限速 部分功能未实现 1 中午吃啥队 点餐系统需求高,系统功能完善 界面可以再完善 2 小谷围驻广东某工业719电竞大队 项目贴近大学生 ...
- [转] KVM虚拟化技术生态环境介绍
KVM虚拟化技术生态环境介绍 http://xanpeng.github.io/wiki/virt/kvm-virtulization-echosystem-intro.html kvm和qemu/q ...
- [转]深入理解 GRE tunnel
我以前写过一篇介绍 tunnel 的文章,只是做了大体的介绍.里面多数 tunnel 是很容易理解的,因为它们多是一对一的,换句话说,是直接从一端到另一端.比如 IPv6 over IPv4 的 tu ...
- 背水一战 Windows 10 (99) - 关联启动: 关联指定的文件类型, 关联指定的协议
[源码下载] 背水一战 Windows 10 (99) - 关联启动: 关联指定的文件类型, 关联指定的协议 作者:webabcd 介绍背水一战 Windows 10 之 关联启动 关联指定的文件类型 ...