神奇的i++

i++,++i,多简单啊,不需要深入研究吧!!!

我是这样想的。

直到我做了一道Java基础检测题,才发现,哦,原来是这样啊!!!

题是这样的

public class Demo {
public static void main(String args[]) {
int num = 50 ;
num = num ++ * 2 ;
System.out.println(num) ;
}
}
最终的执行结果是什么?

结果是什么?100?102?101?

正确输出是:100

对了就不必往下看了.

Javap介绍

先来看个有用的指令:javap

javap:

javap是jdk自带的反解析工具,通过反编译生成的汇编代码,我们可以深入的了解java代码的工作机制

javap的用法格式:

javap

classes就是你要反编译的class文件

D:\notepadd++cache>javap
用法: javap <options> <classes>
其中, 可能的选项包括:
-help --help -? 输出此用法消息
-version 版本信息
-v -verbose 输出附加信息
-l 输出行号和本地变量表
-public 仅显示公共类和成员
-protected 显示受保护的/公共类和成员
-package 显示程序包/受保护的/公共类
和成员 (默认)
-p -private 显示所有类和成员
-c 对代码进行反汇编
-s 输出内部类型签名
-sysinfo 显示正在处理的类的
系统信息 (路径, 大小, 日期, MD5 散列)
-constants 显示最终常量
-classpath <path> 指定查找用户类文件的位置
-cp <path> 指定查找用户类文件的位置
-bootclasspath <path> 覆盖引导类文件的位置

Jvm指令介绍

这是接下来要用到的Jvm指令的含义,推荐一篇JVM指令博客JVM指令介绍博客

//指令含义

bipush : valuebyte值带符号扩展成int值入栈

istore : 将栈顶int类型值保存到局部变量indexbyte中

iinc : 将指定int型变量增加指定值(i++, i--, i+=2)

iload : 从局部变量indexbyte中装载int类型值入栈

iconst_2: 2(int)值入栈

imul : 将栈顶两int类型数相乘,结果入栈。

分析

通过javap来分析下程序的运行机制

把这段代码放到StrTest.java中

 public class StrTest {
public static void main(String args[]) {
int num1 = 50 ;
num1 = num1++ * 2 ;
System.out.println(num1) ;
}
}

通过cmd执行javac StrTest.java ->javap -c StrTest.class得到如下反汇编代码

public class StrTest {
public StrTest();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public static void main(java.lang.String[]);
Code:
0: bipush 50
2: istore_1
3: iload_1
4: iinc 1, 1
7: iconst_2
8: imul
9: istore_1
10: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
13: iload_1
14: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
17: return
}

把上边的内容添加上注释,瞬间就能明白,为啥结果是100了。

public static void main(java.lang.String[]);
Code:
0: bipush 50 //把50推入栈顶top
2: istore_1 //将50保存到第二个局部变量中->num1=50
3: iload_1 //把局部变量num1的值推入栈顶,此时栈顶值top是50
4: iinc 1, 1 //将50执行加1->50+1 局部变量num1值是51
7: iconst_2 //把2推入栈顶
8: imul //把50和2相乘,并把结果100推入栈顶,此时栈顶值top是100
9: istore_1 //把100保存到第二个局部变量中->num1=100
10: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
13: iload_1
14: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
17: return

通过Jvm指令解析后,可知程序经过的步骤是这样的:

  1. 把50推入栈顶,栈顶值是50
  2. num1的值是50
  3. num1执行自增->num1=51
  4. 把2推入栈顶,栈顶值是2
  5. 执行50*2并保存到栈顶,栈顶值是100
  6. 把栈顶值赋给num1->num1=100

同理,++i也好理解了,把上边程序的num1++改为++num1

结果102,再分析下执行步骤

 public static void main(java.lang.String[]);
Code:
0: bipush 50 //把50推入栈顶top
2: istore_1 //将50保存到第二个局部变量中->num1=50
3: iinc 1, 1 //将50执行加1->50+1 局部变量num1值是51
6: iload_1 //把局部变量num1的值推入栈顶,此时栈顶值top是51
7: iconst_2 //把2推入栈顶
8: imul //把51和2相乘,并把结果102推入栈顶,此时栈顶值top是102
9: istore_1 //把102保存到第二个局部变量中->num1=102
10: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
13: iload_1
14: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
17: return

通过Jvm指令解析后,可知程序经过的步骤是这样的:

  1. 把50推入栈顶,栈顶值是50
  2. num1的值是50
  3. num1执行自增->num1=51
  4. 把num1的值推入栈顶,栈顶的值是51
  5. 把2推入栈顶,栈顶值是2
  6. 执行51*2并保存到栈顶,栈顶值是102
  7. 把栈顶值赋给num1->num1=102

结束

ok,分析结束,又温习了一遍指令,记录一下

神奇的i++的更多相关文章

  1. BZOJ 1006 【HNOI2008】 神奇的国度

    题目链接:神奇的国度 一篇论文题--神奇的弦图,神奇的MCS-- 感觉我没有什么需要多说的,这里简单介绍一下MCS: 我们给每个点记录一个权值,从后往前依次确定完美消除序列中的点,每次选择权值最大的一 ...

  2. 前端精选文摘:BFC 神奇背后的原理

    BFC 已经是一个耳听熟闻的词语了,网上有许多关于 BFC 的文章,介绍了如何触发 BFC 以及 BFC 的一些用处(如清浮动,防止 margin 重叠等).虽然我知道如何利用 BFC 解决这些问题, ...

  3. MVC系列——MVC源码学习:打造自己的MVC框架(四:了解神奇的视图引擎)

    前言:通过之前的三篇介绍,我们基本上完成了从请求发出到路由匹配.再到控制器的激活,再到Action的执行这些个过程.今天还是趁热打铁,将我们的View也来完善下,也让整个系列相对完整,博主不希望烂尾. ...

  4. 一行神奇的javascript代码

    写本篇文章的缘由是之前群里@墨尘发了一段js代码,如下: (!(~+[])+{})[--[~+""][+[]]*[~+[]] + ~~!+[]]+({}+[])[[~!+[]]*~ ...

  5. [翻译svg教程]Path元素 svg中最神奇的元素!

    先看一个实例 <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999 ...

  6. php isset( $test ) 的神奇之处。

    很久一段时间没更新博客了,由于近段时间一直在忙 挑战杯 的项目,所以没怎样把一些总结放上来.这次,总结下 php 的一个 函数 : boolean isset($test), 返回值:boolean类 ...

  7. 神奇的CSS3按钮特效

    点击这里查看效果 以下是源代码: <!doctype html> <html> <!-- author: @simurai --> <head> < ...

  8. 吉特仓库管理系统- 斑马打印机 ZPL语言的腐朽和神奇

    上一篇文章说到了.NET中的打印机,在PrintDocument类也暴露一些本质上上的问题,前面也提到过了,虽然使用PrintDcoument打印很方便.对应条码打印机比如斑马等切刀指令,不依赖打印机 ...

  9. JS开发HTML5游戏《神奇的六边形》(一)

    近期出现一款魔性的消除类HTML5游戏<神奇的六边形>,今天我们一起来看看如何通过开源免费的青瓷引擎(www.zuoyouxi.com)来实现这款游戏. (点击图片可进入游戏体验) 因内容 ...

  10. 神奇的sort()函数

    今天来谈一谈sort()函数,sort() 方法用于对数组的元素进行排序,用法为arrayObject.sort(sortby):括号中的为可选参数,准确来说应该是一个函数,这个函数用来规定排序方法, ...

随机推荐

  1. 十四 OGNL的概述

    1 什么是OGNL? 对象图导航语言(对象图,以任意一个对象为根,通过OGNL;可以访问与这个对象关联的其他对象).比EL表达式强大很多倍. 可以存取对象的任意属性.调用对象的方法,遍历整个对象的结构 ...

  2. 吴裕雄 Bootstrap 前端框架开发——Bootstrap 表格:联合使用所有表格类

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  3. B: 最小代价

    B: 最小代价 题解:先用最小生成树求联通所有点的最小代价ans 在求度为1的时候权值最大的点mx ans-mx就是答案 #include<iostream> #include<al ...

  4. hexo 安装和部署

    因为投递论文需要个人网站,所以今天又来了一次hexo 安装和部署. 参考官方文档:https://hexo.io/zh-cn/docs/ git是需要的哈.自行安装 node需要大于8.10 node ...

  5. Aho-Corasick (AC) 自动机

    基础:AC自动机是建立在 trie 树和 kmp 基础之上的,为什么这么说,因为AC自动机是基于字典树的数据结构之上的,其次它是一个自动机,用到了 kmp 的失配数组的思想. 应用:在模式匹配的问题中 ...

  6. Android之收音机UI实现(转)

    源码: http://www.2cto.com/kf/201211/171417.html 最近在研究收音机的源码,本来想把收音机从源码中提取出来,做成一个单独的应用,但是,收音机需要底层的支持,所以 ...

  7. UCOS-III API函数

    附录:UCOS-III API函数 任务管理 就绪列表 挂起队列 时间管理 信号量 消息队列 内存管理

  8. pytest+allure(pytest-allure-adaptor基于这个插件)设计定制化报告

    一:环境准备 1.python3.6 2.windows环境 3.pycharm 4.pytest-allure-adaptor 5.allure2.8.0 6.java1.8 pytest-allu ...

  9. zabbix通过ipmi传感器监控浪潮服务器的硬件信息

    一:实验对象 操作系统版本:centos7.6 监控对象:通过服务器传感器获取到的所有在使用的硬件信息 zabbix版本: 4.0.14二:zabbix介绍 zabbix适合中小型企业.大型企业的用户 ...

  10. python字符串和数值操作函数大全(非常全)

    字符串和数值型数字的操作大全 1.反斜杠\的使用规则:一般使用表示续行的操作,可以其他符号相结合组成其他的一些使用符号,转义字符\‘的使用会识别引号,使得字符串中的引号和外面本来的啊引号相区分. (1 ...