java中j 和 j啥区别_从字节码层次分析++j和j++的区别

一、缘起

最近看到个面试题:

int j = 0;

for(int i = 0; i <100; i++)

j = j++;

System.out.println(j);

输出结果是0,如果换成j++,那么输出是100,这是为什么呢?

二、++的运算区别

对于这种问题,其实有点经验的程序员都知道,前置++和后置++的运算区别:

1.前置++是将自身加1的赋值给新变量,同时自身也加1

2.后置++是将自身的赋值给新变量,然后才自身加1

话虽如此,那如何来证明上述的两点结论呢?

万物皆有始终,对于代码,它的始就是字节码。

在讲字节码之前,先简单回顾一下Java栈,在JVM中有这么一个数据结构叫Java栈,当线程启动的时候,会分配一块内存当做该线程的栈,每个栈由一系列的栈帧组成。每个栈帧对应一个方法,当线程执行方法时,就是栈帧出栈入栈的过程。

每个栈帧包含三部分数据:本地变量(参数+方法内的变量)、操作数栈和其他数据,本文主要涉及本地变量和操作数栈。

1f1745a0cc9c?utm_campaign

栈帧

先看看后置++的实现:

public static void main(String[] args) {

int i = 0;

i = i++;

}

编译之后的字节码如下:

1f1745a0cc9c?utm_campaign

字节码

iconst_0 //把数值0 push到操作数栈

istore_1 //把操作数栈写回到本地变量第2个位置

iload_1 //把本地变量第2个位置的值push到操作数栈

iinc 1,1 //把本地变量表第2个位置加1

istore_1 //把操作数据栈写回本地变量第2个位置

整个过程这样的:

1f1745a0cc9c?utm_campaign

运行过程

可以发现变量在执行iinc1,1的时候已经变成1了,但是istrore_1又把变量a所在位置覆盖成了0,所以执行完i = i++,i还是原来那个值。

接着,来看下前置++i的实现:

1f1745a0cc9c?utm_campaign

前置++的实现

整个过程实现如下:

1f1745a0cc9c?utm_campaign

执行过程

和后置++不同的地方发在于,在变量进入操作数栈之前,就先执行了iinc指令,所以进入操作数的值是加1后的值,最后写回的值就是加1后的新值了。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值