也许我这是在较真, 但是我们确实有时候就不小心就错写为这种情况了。

看如下代码:

public class Test{

        public static void main(String[] args){
                int a = 3;
                int b = 5;
                a = a++;
                b = ++b;
        }

}

  这时候, 如果输出 a 和 b ,那么 他们的值是什么? 答案是 a = 3; b = 6;

  如果你感到迷惑, 那么继续往下看, 如果你知道其中的原理,那么就不用看了。

我们利用 jdk 自带的两个命令(javac,javap)来看看 JVM 虚拟机到底在底层做了些什么。

建立一个文件 Test.java 并把上面的代码写入其中, 然后里用 javac 命令编译

wuqinglong@debian:~$ javac -g:vars Test.java

  然后利用 javap 命令查看 JVM 编译之后的虚拟机指令。

wuqinglong@debian:~$ javap -verbose Test

  输出如下内容

Classfile /home/wuqinglong/Test.class
  Last modified 2016-3-16; size 326 bytes
  MD5 checksum 2e2f3ec62c26d0e1e53bb121e83b9032
public class Test
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#17         // java/lang/Object."<init>":()V
   #2 = Class              #18            // Test
   #3 = Class              #19            // java/lang/Object
   #4 = Utf8               <init>
   #5 = Utf8               ()V
   #6 = Utf8               Code
   #7 = Utf8               LocalVariableTable
   #8 = Utf8               this
   #9 = Utf8               LTest;
  #10 = Utf8               main
  #11 = Utf8               ([Ljava/lang/String;)V
  #12 = Utf8               args
  #13 = Utf8               [Ljava/lang/String;
  #14 = Utf8               a
  #15 = Utf8               I
  #16 = Utf8               b
  #17 = NameAndType        #4:#5          // "<init>":()V
  #18 = Utf8               Test
  #19 = Utf8               java/lang/Object
{
  public Test();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   LTest;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=1, locals=3, args_size=1
         0: iconst_3    // 将int类型的3送到栈顶
         1: istore_1    // 将栈顶的int类型的值存储到slot为1的int类型的本地变量中,slot编号对应的变量名在后三行。
         2: iconst_5    // 将int类型的5送到栈顶
         3: istore_2    // 将栈顶的int类型的值存储到slot为2的int类型的本地变量中
         4: iload_1    // 将slot为1的int类型的本地变量加载到栈顶
         5: iinc          1, 1    // 将slot为1的本地int类型的变量的值加1
         8: istore_1    // 将栈顶的int类型的值存储到slot为1的本地变量中
         9: iinc          2, 1    // 将slot为2的本地intbeijing的变量的值加1
        12: iload_2    // 将slot为2的int类型的本地变量加载到栈顶
        13: istore_2    // 将栈顶的int类型的值存储到slot为2的int类型的本地变量中
        14: return    // 结束
      LocalVariableTable:    // 变量对应的slot编号如下
        Start  Length  Slot  Name   Signature
            0      15     0  args   [Ljava/lang/String;
            2      13     1     a   I
            4      11     2     b   I
}

  如果有点汇编知识的话,看上面的代码会比较好理解点, 如果没有学过汇编指令的话, 就看注释。

其中最主要的区别是:

  对于  a = a++;  JVM 进行的操作是:

    1. 将本地变量的值加载到栈顶  (注意: 这时候这个值是没有经过自增操作的)

    2. 对本地变量的值进行自增(对栈顶对象没有影响)

    3. 把栈顶变量的值存储到本地变量中(重新对本地变量赋值,栈顶变量的值并没有进行自增操作,自增操作是对本地变量进行操作的)

  对于  b = ++b;  JVM 进行的操作是:

    1. 将本地变量的值先自增

    2. 将自增完之后的本地变量的值加载到栈顶(注意:这时候这个值是经过了自增操作的)

    3. 将栈顶变量的值存储到本地变量中

Java中的自增问题(i=i++)的更多相关文章

  1. 【Java】【4】关于Java中的自增自减

    摘要:理解j = j++与j = ++j的区别:正确用法:直接用j++,不要用前两种 正文: import java.util.*; public class Test{ public static ...

  2. java中的自增问题

    运行下面这段代码,其结果是什么呢? package com.test; public class Inc { public static void main(String[] args) { Inc ...

  3. java中集合的增删改操作及遍历总结

      集合的增删改操作及遍历总结

  4. java中双向链表的增、删、查操作

    import java.util.NoSuchElementException; public class DoublyLinkedListImpl<E> { private Node h ...

  5. Java中的自增自减

    情况①: for (int i = 0; i < 100; i++) { j = 1 + j++; } System.out.println(j); 结果是 0 !! 这是由于在进行后自增/自减 ...

  6. Java中AtomicInteger的使用!!!

    今天在看Volley的源码的时候,看到里面使用了AtomicInteger这个类,曾经没用过,今天看了一下API学习了一下: 首先介绍一下这个类的用处,这个类主要是用来替换java中的自增和自减操作, ...

  7. Android(java)学习笔记186:对ListView等列表组件中数据进行增、删、改操作

    1.ListView介绍 解决大量的相似的数据显示问题 采用了MVC模式: M: model (数据模型) V:  view  (显示的视图) C: controller 控制器 入门案例: acit ...

  8. Android(java)学习笔记129:对ListView等列表组件中数据进行增、删、改操作

    1. ListView介绍 解决大量的相似的数据显示问题 采用了MVC模式: M: model (数据模型) V:  view  (显示的视图) C: controller 控制器 入门案例: aci ...

  9. java中如何理解:其他类型 + string 与 自增类型转换和赋值类型转换

    java中如何理解:其他类型 + string 与 自增类型转换和赋值类型转换 一.字符串与其他类型连接 public class DemoString{ public static void mai ...

随机推荐

  1. HDU2896 病毒侵袭

    题目大意:给出若干病毒的特征码,不超过500个.每个病毒的特征码长度在20~200之间.现在有若干网站的源代码,要检测网站的源代码中是否包含病毒.网站的个数不超过1000个,每个网站的源代码长度在70 ...

  2. sql:劳务统计各分公司管理费用明细合计(等同汇总报表)

    select gl_balance.year, bd_accsubj.dispname, sum(gl_balance.debitamount) 收入, sum(gl_balance.creditam ...

  3. LINUX 下Open cv练习使用小记(1)

    首先肯定离不开选一张自己喜欢的图像来显示 #include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp ...

  4. js加解密

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  5. web应用虚拟目录的映射

    1.新建一个web应用.在D盘新建一个news文件夹,文件夹下面新建一个html文件.如下图所示: 2.将web应用映射到服务器的虚拟目录 第一种方式: 2.1 用记事本打开tomcat目录下面的se ...

  6. Repeater控件使用中的一些小问题

    网页上用来展示列表的数据,发现还是Repeater比GridView,DetailView之类的要灵活些,所以近期用到了就总结下遇到的一些情况,保留下来以备之后查阅,不用现问度娘了... 自己摸索的, ...

  7. 删除sqlserver代理任务脚本

    无法删除SQLSERVER代理任务时可用如下语句试验下 use [msdb] ) set @job_name = N'jobname' --注:job_name为维护计划对应的job name --删 ...

  8. 慕课网-Java入门第一季-6-7 使用 Arrays 类操作 Java 中的数组

    来源:http://www.imooc.com/code/1556 Arrays 类是 Java 中提供的一个工具类,在 java.util 包中.该类中包含了一些方法用来直接操作数组,比如可直接实现 ...

  9. Jmeter模拟不同带宽

    Jmeter自带模拟带宽设置,当然前提肯定是你当前的带宽>=你要模拟的带宽,好比你装了个4m的宽带,要模拟100m的带宽,那是做梦 做起来也不难,打开user.properties文件,增加如下 ...

  10. vi 使用入门

    几种模式:     Normal Mode 命令模式     Insert Mode 编辑模式     Command-line Mode      Visual Mode      Select M ...