从字节码层次看i++和++i

Posted alinainai

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从字节码层次看i++和++i相关的知识,希望对你有一定的参考价值。

  技术图片

  关于的Java的i++和++i的区别,初学者可能会混淆,这时候有经验的同学或同事就会告诉你,++在后,就会立马加值,

++在后则会等会儿再加,所以如果i == 0 ,那么i++ == 0,++i == 1。

  那么这个先加后加具体在字节码中是怎样一个逻辑呢?这个就需要我们去看看Java的字节码了,如何查看字节码请

参考我之前的文章IDEA设置External Tools之Javap反编译字节码

  准备两个函数

  函数test1

  技术图片

  函数test2

  技术图片

  先直接上函数的打印结果就,再来慢慢分析

  技术图片

   先看test1的字节码

  技术图片

  0~3行:一次定义了i和j两个变量,并赋值为0,存放在局部变量表。

  4~7行:开始循环的逻辑,if_icmpge是比较语句,如果j>=50则直接跳到第21行,调用打印方法。

  核心的逻辑在10~15行:

  技术图片

   现将局部变量表slot 0的变量加载到操作数栈,接着对局部变量表自增(没错,你没看错,这里是对局部

变量表自增而不是对操作数栈的变量自增)。。紧接着下一步istore_0又把栈顶的值存回局部变量表slot_0,所以,。。

最终局部变量表的值是没有改变的。

  i++分析完毕,再回看上面的字节码,其中还有值得注意的地方:

  技术图片

   21~43行:这段对应代码

1 System.out.println("i++: " + i);

  先new一个PrintStream对象,下面可以看到是new了一个StringBuilder对象。这里可能是jvm的一个优化,将字符串拼接

改成了StringBuilder来append,所以我们经常看到的有人说要把字符串拼接改成StringBuilder其实大可不必,先看下字节码

对于语句的优化是怎样的。

  接着来看test2方法的字节码:

技术图片

   前面的代码和test1一样,我们直接来看不一样的地方:

  技术图片

   该段字节码对应代码:

i = ++i;

  先对局部变量表上的值自增,再加到操作数栈,最终存回到局部变量表,这样的话变量表的值就是实实在在

增加了的。

  所以,回想上面的分析,得出结论就是:i++是将变量表的值加载操作数栈后再对变量表上的值自增,而++i

则是先对变量表上的值自增后再将其加载到操作数栈。

  当然,殊途同归,最终不管是i++还是++i,都是为了自增。

 

  ~~本文结束,感谢各位看官。

  

 

以上是关于从字节码层次看i++和++i的主要内容,如果未能解决你的问题,请参考以下文章

java字节码角度图解 i++ 和 ++i

JVMJVM05(从字节码角度分析i++和++i的执行流程)

java中i=i++字节码分析

字节码分析finally块对return返回值的影响

java字节码I++ ++j

从 JVM 层面理解 i++ 和 ++i 的真正区别!