java的++和--操作符
只要是会java的都知道++和—操作符的用法,如
- int i = 1;
- int j = i++;
- int k = ++i;
结果i为3,j为1,k为3。
那如下代码:
- int j = 0;
- for (int i = 0; i < 100; i++) {
- j = j++;
- }
- System.out.println(j);
输出结果又是多少呢?100?0?
正确答案是0。为什么呢?
要想搞明白这个问题,那来看看这段代码生成的字节码:
- 0: iconst_0
- 1: istore_1
- 2: iconst_0
- 3: istore_2
- 4: goto 15
- 7: iload_1
- 8: iinc 1, 1
- 11: istore_1
- 12: iinc 2, 1
- 15: iload_2
- 16: bipush 100
- 18: if_icmplt 7
- 21: getstatic #16; //Field java/lang/System.out:Ljava/io/PrintStream;
- 24: iload_1
- 25: invokevirtual #22; //Method java/io/PrintStream.println:(I)V
- 28: return
j = j++;的字节码为:
- 7: iload_1
- 8: iinc 1, 1
- 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。如:
- int x = 1;
- x = x + -128;
- x = x + -129;
- x = x + 127;
- x = x + 128;
生成字节码如下:
- 0: iconst_1
- 1: istore_1
- 2: iinc 1, -128 //x = x + -128;
- 5: iinc_w 1, -129 //x = x + -129;
- 11: iinc 1, 127 //x = x + 127;
- 14: iinc_w 1, 128 //x = x + 128;
- 20: return
上面把innc指令的基本意思说清楚了,现在是最重要的一点:
innc指令操作的是局部变量表中的变量,而不是当前操作数栈栈顶的数据(iinc指令实现有没有用到操作数栈已经不重要了)。
上面j = j++;的字节码
- 7: iload_1
- 8: iinc 1, 1
- 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,再把局部变量取出来放到操作数栈顶。如下代码:
- int j = 0;
- int x = j++;
- x = ++j;
字节码如下:
- 0: iconst_0 //把常量0放到操作数栈顶
- 1: istore_1 //int j = 0;
- 2: iload_1 //先取出j的值0
- 3: iinc 1, 1 //j++,把局部变量表中j的值加1,j=1,此时操作数栈顶的值还是0
- 6: istore_2 //把操作数栈顶的值0放加局部变量表中,此时x=0
- 7: iinc 1, 1 //++j,先把局部变量表中j的值加1,此时j=2
- 10: iload_1 //取出j的值2
- 11: istore_2 //x=2
- 12: return //方法返回
以上说的都是++操作符,--的操作也是一样的,就不再多说。
java的++和--操作符的更多相关文章
- Java的常用操作符
操作符用于接受一个或多个参数,并生成一个新的值.加号和一元的正号(+).减号和一元的负号(一).乘号(*).除号(/)以及赋值号(=)的用法和大多数编程语言都类似. 操作符作用于操作数,生成一个新值. ...
- java 语法错误 (操作符丢失) 在查询表达式
遇到的详细问题: a[0]="11"; a[1]="2223"; a[2]="333"; sta.executeUpdate("i ...
- java学习 之 操作符
操作符介绍 java语言操作符 1.赋值操作符 = 2.计算操作符 + .- (减.负号).*(乘)./(除) 3.递增递减 --(递减).++(递增) 4.关系操作符 ==.!= ...
- Java中的“==操作符”和equals方法有什么区别
Java中的"=="和equals方法究竟有什么区别? 1.==操作符 "=="操作符专门用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中所存储的 ...
- Java数据类型、操作符、表达式
基本与C#相同,因C#从Java学的 如操作符 对象的Equals方法,比较两个对象的内容是否相等. ==是比较是否引用同一对象.
- java中的==操作符和equals函数
基本规则 “==”操作符的使用需要分成两种情况 判值类型相等 这一点很好理解,两个值类型代表的数值相等,则“==”表达式返回true “==”可以用与不同值类型的比较,语言会自动进行类型转换 判引用类 ...
- Java基础(一)--操作符
Java底层都是使用操作符来操作Java中的数据 常见的操作符:+.-.*./.= 优先级: 当一个表达式存在多个操作符时,操作符的优先级决定了计算顺序,这点在我们刚开始学习数学的时候就会了解到 如果 ...
- Thinking In Java 4th Chap3 操作符
若String后接一‘+’运算符,其后元素自动转化为String类型 注意:若对对象赋值另一对象,操作对应的是引用,如c=d,则c和d都指向原来d指向的对象 生成随机数:Random rand=new ...
- C++ 操作符重载实践 & java没有重载操作符的思路
实践如下: #include <iostream> using namespace std; class Book{ private: int page; public: Book(int ...
随机推荐
- Bilateral Filtering(双边滤波) for SSAO
原网址:http://blog.csdn.net/bugrunner/article/details/7170471 1. 简介 图像平滑是一个重要的操作,而且有多种成熟的算法.这里主要简单介绍一下B ...
- 「小程序JAVA实战」开发用户redis-session(40)
转自:https://idig8.com/2018/09/05/xiaochengxujavashizhankaifayonghuredis-session39/ 接下来我们需要在我们的项目里面配置下 ...
- Spring cloud Eureka高可用 - Windows 7 hosts文件立即生效
hosts 文件所在位置 c:/windows/system32/drivers/etc/hosts 左下角 搜索框 搜索 cmd 弹出命令框 输入 ipconfig /displaydns 显示所有 ...
- 安装Apache2
Linux下安装Apache 2.4 2012-08-06 09:36:51| 分类: linux|字号 订阅 本文原创,欢迎转载.转载请在文章明显可见处张贴如下内容:(注意:请保留超链接格 ...
- effective javascript 学习心得
第2条:理解JavaScript的浮点数 1.js数字只有一种类型 2.见代码 /** * Created by Administrator on 2017/7/2. */ console.log(& ...
- 关于IOS新手在安装cocoa pods失败,因为ruby版本过低的解决方法+ (void) {升级ruby}
http://blog.csdn.net/zhaoen95/article/details/51995520 现在: 使用 OS 系统, 正在学习iOS 需要配置cocoapods 命令行中显 ...
- c# 类中使用ResolveUrl
类中使用ResolveUrl 1>获取当前page然后调用ResolveUrl System.Web.UI.Page page = HttpContext.Current.CurrentHand ...
- linux操作系统下,怎么使用kill按照PID一次杀死多个进程
1.ps -ef | grep firefox | grep -v grep | cut -c 9-15 | xargs kill -s 9 说明:“grep firefox”的输出结果是,所有含有关 ...
- Loadrunner11无法在win7 64位上启用ie解决办法
Loadrunner11无法在win7 64位上启用ie解决办法 1.loadrunner11在win7 64位上默认启用的是32位的那个IE浏览器,路径:C:\Program Files (x86) ...
- linux 下删除乱码文件-乾颐堂
在linux下删除文件,遇到特殊字符是一件非常头疼的事情. 1. 如果文件名带 ‘-’ 或者‘--’这样的字符 删除办法为:rm -- 文件名 如文件名为:-pythontab.tgz 如果用普通方法 ...