今天偶然呗问到了一个java中自增运算符的问题,涉及到了堆栈,觉得挺有趣的把它记录下来。

int i=0;
      i=i++;
      结果i是多少?
      这是一个经常被提及的问题,答案一直五花八门。
      具体测试一下以说明问题:
      代码1:

  1. public class Test{
  2. public static void main(String[] args){
  3. int i=0;
  4. i=i++;
  5. System.out.println(i);
  6. }
  7. }

结果i依然是0.分析其反编译后的代码:

  1. public static void main(java.lang.String[]);
  2. Code:
  3. 0:   iconst_0     //0放到栈顶
  4. 1:   istore_1    //把栈顶的值保存到局部变量1,也就是i中
  5. 2:   iload_1     //把i的值放到栈顶,也就是说此时栈顶的值是0
  6. 3:   iinc    1, 1  //注意这个指令,把局部变量1,也就是i,增加1,这个指令不会导致栈的变化,也就是说局部变量1,即i此时为1了。
  7. 6:   istore_1     //把栈顶的值(0)保存到局部变量1,也就是让i为0了,所以最后i为0
  8. 7:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;
  9. 10:  iload_1
  10. 11:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V
  11. 14:  return

值得注意到是i被修改了两次,第一次是i++;i变为1,最后一次是i=0;所以结果i是0
代码2:

  1. public class Test2{
  2. public static void main(String[] args){
  3. int i=0;
  4. int j=0;
  5. j=i++;
  6. System.out.println(i);
  7. System.out.println(j);
  8. }
  9. }

这个结果肯定都知道,i是1,j是0.同样看反编译之后的代码:

  1. public static void main(java.lang.String[]);
  2. Code:
  3. 0:   iconst_0
  4. 1:   istore_1     //i=0
  5. 2:   iconst_0
  6. 3:   istore_2     //j=0
  7. 4:   iload_1      //把i的值放到栈顶,也就是说此时栈顶的值是0
  8. 5:   iinc    1, 1  //局部变量1加1,也就是让i++了,此时i已经是1了,上面说过,此指令不会导致栈变化
  9. 8:   istore_2     //把栈顶的值(注意是0)存入局部变量2,也就是j中,所以j=0
  10. 9:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;
  11. 12:  iload_1
  12. 13:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V
  13. 16:  getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;
  14. 19:  iload_2
  15. 20:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V
  16. 23:  return

很明显可以看出,java是先把i的值取出来放到栈顶,我们可以认为是引入了第三个变量int k=i;然后i++,这时候i为1了,然后让j=k;也就是0.结论,i的++运算是在对j这个变量的赋值之前完成的。

代码3:

  1. public class Test3{
  2. public static void main(String[] args){
  3. int i=0;
  4. int j=0;
  5. j=++i;
  6. System.out.println(i);
  7. System.out.println(j);
  8. }
  9. }

结果大家也都知道,i=1,j=1
看操作过程:

  1. public static void main(java.lang.String[]);
  2. Code:
  3. 0:   iconst_0
  4. 1:   istore_1    //i=0
  5. 2:   iconst_0
  6. 3:   istore_2     //j=0
  7. 4:   iinc    1, 1   //局部变量i加1,这时候i变成1了 。
  8. 7:   iload_1     //把i的值放到栈顶,栈顶的值是1
  9. 8:   istore_2    //j=1
  10. 9:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;
  11. 12:  iload_1
  12. 13:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V
  13. 16:  getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;
  14. 19:  iload_2
  15. 20:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V
  16. 23:  return

对比代码2和代码3,关键的差别就是iload_1   个iinc这两条指令的位置变了。

java中i = i++问题的更多相关文章

  1. java中的锁

    java中有哪些锁 这个问题在我看了一遍<java并发编程>后尽然无法回答,说明自己对于锁的概念了解的不够.于是再次翻看了一下书里的内容,突然有点打开脑门的感觉.看来确实是要学习的最好方式 ...

  2. java中的字符串相关知识整理

    字符串为什么这么重要 写了多年java的开发应该对String不陌生,但是我却越发觉得它陌生.每学一门编程语言就会与字符串这个关键词打不少交道.看来它真的很重要. 字符串就是一系列的字符组合的串,如果 ...

  3. Java中的Socket的用法

                                   Java中的Socket的用法 Java中的Socket分为普通的Socket和NioSocket. 普通Socket的用法 Java中的 ...

  4. java中Action层、Service层和Dao层的功能区分

    Action/Service/DAO简介: Action是管理业务(Service)调度和管理跳转的. Service是管理具体的功能的. Action只负责管理,而Service负责实施. DAO只 ...

  5. Java中常用集合操作

    一.Map 名值对存储的. 常用派生类HashMap类 添加: put(key,value)往集合里添加数据 删除: clear()删除所有 remove(key)清除单个,根据k来找 获取: siz ...

  6. java中的移位运算符:<<,>>,>>>总结

    java中有三种移位运算符 <<      :     左移运算符,num << 1,相当于num乘以2 >>      :     右移运算符,num >& ...

  7. 关于Java中进程和线程的详解

    一.进程:是程序的一次动态执行,它对应着从代码加载,执行至执行完毕的一个完整的过程,是一个动态的实体,它有自己的生命 周期.它因创建而产生,因调度而运行,因等待资源或事件而被处于等待状态,因完成任务而 ...

  8. Java中的进程和线程

     Java中的进程与线程 一:进程与线程 概述:几乎任何的操作系统都支持运行多个任务,通常一个任务就是一个程序,而一个程序就是一个进程.当一个进程运行时,内部可能包括多个顺序执行流,每个顺序执行流就是 ...

  9. Java中的进程与线程(总结篇)

    详细文档: Java中的进程与线程.rar 474KB 1/7/2017 6:21:15 PM 概述: 几乎任何的操作系统都支持运行多个任务,通常一个任务就是一个程序,而一个程序就是一个进程.当一个进 ...

  10. 初探java中this的用法

    一般this在各类语言中都表示“调用当前函数的对象”,java中也存在这种用法: public class Leaf { int i = 0; Leaf increment(){ i++; retur ...

随机推荐

  1. Git使用总结(一):简介与基本操作

    一:简介 GIT是一个开源的分布式的版本控制系统,是由Linus 为了管理Linux内核开发而开发的一个开源的版本控制软件.相比SVN,它采用分布式版本库方式. 二:工作区,暂存区和版本库 左侧为工作 ...

  2. SpringMVC Controller层的单元测试

    Getting Ready 测试相关Maven dependency如下: <dependency> <groupId>org.springframework</grou ...

  3. JS原型与原型链图解

  4. 「Sdchr 的邀请赛」题解

    骗个访问量.. A:取石子 将点 x 与点 x / prime 连边,那么这个图可以由指数之和的奇偶性来划分成一个二分图. 接下来考虑推广阶梯 NIM (或者这原本就是阶梯 NIM ?),必胜当且仅当 ...

  5. kd-tree 小结

    核心思想 是一种分割 \(k\) 维数据空间的数据结构 一维情况下就是平衡树,以 \(key\) 为标准判断插入左儿子还是右儿子 \(kdtree\) 就是平衡树在多维空间的扩展 因为有多维,我们按不 ...

  6. 记一次java程序内存溢出问题

    一个自然语言处理程序,在封装为web-service后,部署到线上运行. 但最近出现了内存溢出的情况,频繁的out of memory. 先盲目尝试在启动脚本中增加-XX:-UseGCOverhead ...

  7. C++(笔)001.

    1.编程范式 编程范式是指计算机编程的基本风格,C++可容纳多种程度范式,如面向对象编程.泛型编程及传统的过程式编程. 2.与C相比较 C++在C语言的基础上新加的特性如下: a.类和对象.继承 b. ...

  8. 2017年11月27日 C#MDI窗体创建&记事本打印&记事本查找、自动换行

    MDI窗体第一个父窗体 把属性里的IsMdiContainer设置为true就可以了 父窗体连接子窗体 //创建一个新的类,用来连接别的窗体,并且别的窗体为唯一窗体 List<Form> ...

  9. 自己实现async和await

    无意当中看了一些博文,说有人想自己尝试实现基于异步操作的方法: 1)直接使用Task(不说咯,这个是微软给我们的标准实现方法). 2)必须继承INotifyCompletion接口,同时自己实现IsC ...

  10. 《MySQL 基础课程》笔记整理(进阶篇)(未完)

    一.MySQL服务安装及命令使用 安装过程就不写了,毕竟百度经验一大把 MySQL 官方文档 MySQL 参考手册中文版 1.MySQL简介 ​ RDBMS(Relational Database M ...