JMV j = j++ 基于栈的解释器执行过程分析

4 篇文章 0 订阅

0. 前言

  1. 问题来源:
    在这里插入图片描述
  2. 疑问:中间变量缓存机制
  3. 深入理解Java中间变量的缓存机制
    看到这位大佬的测试用例,我简直惊呆了,于是决定自己测试一下,以便从JVM层面了解问题的根源

1. 测试用例:

public class Test02
{
    public static void main(String[] args)
    {
        int i = 8;
        i++;
        System.out.println(i);

        int j = 8;
        j = j++;
        System.out.println(j);
    }
}

2. 代码执行结果

9
8

3. jclasslib 后得到的 main 函数部分的字节码指令

 0 bipush 8
 2 istore_1
 3 iinc 1 by 1
 6 getstatic #2 <java/lang/System.out>
 9 iload_1
10 invokevirtual #3 <java/io/PrintStream.println>
13 bipush 8
15 istore_2
16 iload_2
17 iinc 2 by 1
20 istore_2
21 getstatic #2 <java/lang/System.out>
24 iload_2
25 invokevirtual #3 <java/io/PrintStream.println>
28 return

4. 指令说明:

局部变量表
在这里插入图片描述
bipush:常量入栈指令,int取值范围(-128~127)
istore_1:出栈指令,将操作数栈中的数据保存到局部变量表中1的位置
iload_1:加载指令,将局部变量表中1位置的数据加载到操作数栈中

5. 基于栈的解释器执行过程

  1. i++执行过程过程说明
程序计数器指令说明
0:bipush 8将常量8保存到操作数栈中
2:istore_1取出操作数栈顶元素8放入局部变量表1的位置处
3:iinc 1 by 1在局部变量槽1的位置上自增
9:iload_1从局部变量表中加载1位置上的值到操作数栈顶

在这里插入图片描述

补充说明:3:变量自增
① 在变量槽中自增的说法参考 从字节码理解Java中局部变量的自增/自减
② 执行3:的时候,操作数栈为空,只有局部变量表中有值

  1. j = j++执行过程说明
程序计数器指令说明
13:bipush 8将常量8保存到操作数栈中
15:istore_2取出操作数栈顶元素8放入局部变量表2的位置处
16:iload_2从局部变量表中加载2位置上的值到操作数栈顶
17:iinc 2 by 1在局部变量槽2的位置上自增
20:istore_2取出操作数栈顶元素8放入局部变量表2的位置处
24:iload_2从局部变量表中加载2位置上的值到操作数栈顶

在这里插入图片描述
上述执行过程梳理:
① 在执行17:自增之前,将局部变量表中的数据保存到操作数栈中
② 在局部变量槽中执行自增操作
③ 操作数栈顶元素的值保存到局部变量表中,覆盖掉执行自增后的值
④ 从局部变量表中把被覆盖掉的值加载到操作栈上,以进行后续的输出操作

6. 总结:

两条语句的指令区别:
在这里插入图片描述
问题:
16::为什么要将数据保存到操作数栈中?
20::该行与赋值号=之间的关联?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值