只要是会java的都知道++和—操作符的用法,如

  1. int i = 1;
  2. int j = i++;
  3. int k = ++i;

结果i为3,j为1,k为3。

那如下代码:

  1. int j = 0;
  2. for (int i = 0; i < 100; i++) {
  3. j = j++;
  4. }
  5. System.out.println(j);

输出结果又是多少呢?100?0?

正确答案是0。为什么呢?

要想搞明白这个问题,那来看看这段代码生成的字节码:

  1. 0:   iconst_0
  2. 1:   istore_1
  3. 2:   iconst_0
  4. 3:   istore_2
  5. 4:   goto    15
  6. 7:   iload_1
  7. 8:   iinc    1, 1
  8. 11:  istore_1
  9. 12:  iinc    2, 1
  10. 15:  iload_2
  11. 16:  bipush  100
  12. 18:  if_icmplt       7
  13. 21:  getstatic       #16; //Field java/lang/System.out:Ljava/io/PrintStream;
  14. 24:  iload_1
  15. 25:  invokevirtual   #22; //Method java/io/PrintStream.println:(I)V
  16. 28:  return

j = j++;的字节码为:

  1. 7:   iload_1
  2. 8:   iinc    1, 1
  3. 11:  istore_1

iload_1 意思是把局部变量表中位置1的变量取出来放到操作数栈中;

iinc    1, 1 这是执行j++操作,把局部变量表中位置1的变量加1;

istore_1 这是把操作数栈顶的值弹出放到局部变量表位置1的变量中。

问题就在innc这个指令,指令格式为:

innc vindex const

这个指令只能操作int类型的变量,有两个操作数,第一个操作数vindex指示局部变量在局部表中的位置索引,第二个操作数const表示要相加的整型常量。如

innc 2 100

表示把局部变量表中位置2的变量加上100。

具体的过程是,把局部变量表中位置2的变量取出来,加上100后,然后再放回到局部变量表中。

而const这个常量值的范围是-128~127,如果超出这个范围,使用指令iinc_w。如:

  1. int x = 1;
  2. x = x + -128;
  3. x = x + -129;
  4. x = x + 127;
  5. x = x + 128;

生成字节码如下:

  1. 0:   iconst_1
  2. 1:   istore_1
  3. 2:   iinc    1, -128 //x = x + -128;
  4. 5:   iinc_w 1, -129 //x = x + -129;
  5. 11:  iinc    1, 127 //x = x + 127;
  6. 14:  iinc_w 1, 128 //x = x + 128;
  7. 20:  return

上面把innc指令的基本意思说清楚了,现在是最重要的一点:

innc指令操作的是局部变量表中的变量,而不是当前操作数栈栈顶的数据(iinc指令实现有没有用到操作数栈已经不重要了)。

上面j = j++;的字节码

  1. 7:   iload_1
  2. 8:   iinc    1, 1
  3. 11:  istore_1

iload_1 先把j的值取出来放到栈顶,此时值为0,执行iinc    1, 1这个指令时,是操作的局部变量表中变量(值为0),把它加1,此时值为1,istore_1指令把当前操作数栈顶的值(还是0),又放回局部变量表位置1的变量中,那局部变量表位置1的变量的值又从1变为0了。

所以,不管怎么循环,j的值永远是0。

下面说说i++和++i的问题:

都知道i++是先使用i的值,再把i的值加1;而++i是先把i的值加1,再使用i的值。

但是实际上是怎么回事呢?其实i++和++i都是使用iinc    vindex, 1指令,区别在于,i++是先把局部变量取出来放到操作数栈顶,再把局部变量表中的变量值加1,而++i是先把局部变量表中的变量值加1,再把局部变量取出来放到操作数栈顶。如下代码:

  1. int j = 0;
  2. int x = j++;
  3. x = ++j;

字节码如下:

  1. 0:   iconst_0 //把常量0放到操作数栈顶
  2. 1:   istore_1 //int j = 0;
  3. 2:   iload_1 //先取出j的值0
  4. 3:   iinc    1, 1 //j++,把局部变量表中j的值加1,j=1,此时操作数栈顶的值还是0
  5. 6:   istore_2 //把操作数栈顶的值0放加局部变量表中,此时x=0
  6. 7:   iinc    1, 1 //++j,先把局部变量表中j的值加1,此时j=2
  7. 10:  iload_1 //取出j的值2
  8. 11:  istore_2 //x=2
  9. 12:  return //方法返回

以上说的都是++操作符,--的操作也是一样的,就不再多说。

java的++和--操作符的更多相关文章

  1. Java的常用操作符

    操作符用于接受一个或多个参数,并生成一个新的值.加号和一元的正号(+).减号和一元的负号(一).乘号(*).除号(/)以及赋值号(=)的用法和大多数编程语言都类似. 操作符作用于操作数,生成一个新值. ...

  2. java 语法错误 (操作符丢失) 在查询表达式

    遇到的详细问题: a[0]="11"; a[1]="2223"; a[2]="333"; sta.executeUpdate("i ...

  3. java学习 之 操作符

    操作符介绍 java语言操作符 1.赋值操作符  = 2.计算操作符   + .- (减.负号).*(乘)./(除) 3.递增递减     --(递减).++(递增) 4.关系操作符    ==.!= ...

  4. Java中的“==操作符”和equals方法有什么区别

    Java中的"=="和equals方法究竟有什么区别? 1.==操作符 "=="操作符专门用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中所存储的 ...

  5. Java数据类型、操作符、表达式

    基本与C#相同,因C#从Java学的   如操作符     对象的Equals方法,比较两个对象的内容是否相等.     ==是比较是否引用同一对象.

  6. java中的==操作符和equals函数

    基本规则 “==”操作符的使用需要分成两种情况 判值类型相等 这一点很好理解,两个值类型代表的数值相等,则“==”表达式返回true “==”可以用与不同值类型的比较,语言会自动进行类型转换 判引用类 ...

  7. Java基础(一)--操作符

    Java底层都是使用操作符来操作Java中的数据 常见的操作符:+.-.*./.= 优先级: 当一个表达式存在多个操作符时,操作符的优先级决定了计算顺序,这点在我们刚开始学习数学的时候就会了解到 如果 ...

  8. Thinking In Java 4th Chap3 操作符

    若String后接一‘+’运算符,其后元素自动转化为String类型 注意:若对对象赋值另一对象,操作对应的是引用,如c=d,则c和d都指向原来d指向的对象 生成随机数:Random rand=new ...

  9. C++ 操作符重载实践 & java没有重载操作符的思路

    实践如下: #include <iostream> using namespace std; class Book{ private: int page; public: Book(int ...

随机推荐

  1. Linux配置IP和防火墙

    前言: 刚刚学完了怎么配置Linux IP和防火墙 前来总结. 准备: 需安装的: setup 正文: 安装基础包 yum groupinstall "Base" setup 选择 ...

  2. CentOS7.6安装Git(IUS方式)

    官网下载地址:https://git-scm.com/download/linux 第一步:安装第三方存储库IUS curl https://setup.ius.io | sh 第二步:安装git y ...

  3. C/S软件的自动升级部署

    升级的原理有好几个,首先无非是将现有版本与最新版本作比较,发现最新的则提示用户是否升级.当然也有人用其它属性比较的,例如:文件大小,或者更新日期.而实现的方法呢? 在.Net时代,我们就有了更多的选择 ...

  4. 教你看懂Code128条形码

    首     页 条码控件 条码技术 条码新闻 合作伙伴 联系我们 常见问题 电话:010-84827961 当前位置:条形码控件网 > 条形码控件技术文章 > >正文   教你看懂C ...

  5. 「小程序JAVA实战」小程序的flex布局(22)

    转自:https://idig8.com/2018/08/09/xiaochengxu-chuji-22/ 之前已经把小程序的框架说完了,接下来说说小程序的组件,在说组件之前,先说说布局吧.源码:ht ...

  6. rails 网站跨域

    7down voteaccepted gem install rack-cors Or in your Gemfile: gem 'rack-cors', :require => 'rack/c ...

  7. 【原】Coursera—Andrew Ng机器学习—课程笔记 Lecture 12—Support Vector Machines 支持向量机

    Lecture 12 支持向量机 Support Vector Machines 12.1 优化目标 Optimization Objective 支持向量机(Support Vector Machi ...

  8. unit_2_homework

    随记2018/4/23 # 找元祖中的元素,移除每个元素的空格,并查找以a或A开头,c结尾的所有元素. # 思路:将i取出来,求得li列表中有多少个元素for i in range(len(li)): ...

  9. Navicat修改查询保存路径

    mysql使用navicat查询时有时候会有很多sql语句, ctrl+s时自动保存在C:\Users\Administrator\Documents\Navicat\MySQL\servers\lo ...

  10. 91. Decode Ways反编译字符串

    [抄题]: A message containing letters from A-Z is being encoded to numbers using the following mapping: ...