05.局部变量表与操作数栈

1) 概述

  1. JVM的字节码执行引擎,功能基本就是输入字节码文件,然后对字节码进行解析并处理,最后输出执行的结果。
  2. 实现方式可能有通过解释器直接执行字节码,或者适通过及时编译器产生本地代码,也就是编译执行,当让也可能两者皆有。
    • HotSpot就是两者皆有,使用频率较多的代码JIT动态编译为本地代码,频率较少的就解释执行。

2) 栈帧概述

  1. 栈帧是用于执行JVM进行方法调用和方法执行的数据结构。
  2. 栈帧随着方法调用而创建,随着方法结束而销毁。
  3. 栈帧里面存储了方法的 局部变量操作数栈、动态连接、方法返回地址等信息。

栈帧的概念结构.png

3) 局部变量表

  1. 局部变量表:用来存放方法参数和方法内部定义的局部变量的存储空间。
  2. 以slot为单位,目前一个slot存放32位以内的数据类型
  3. 对于64位的数据占2个slot
  4. 对于实例方法,第0位slot存放的是this,然后从1到n,依次分配给参数列表
    • 对于静态方法,则从0位开始依次分配给参数列表
  5. 然后根据方法体内部定义的变量顺序和作用域来分配slot
public class Test1 {
    public int add(int a, int b) {
        int c = a + b;
        return a + b + c;
        /* javap -verbose 得到slot分配情况
         LocalVariableTable:
           Start  Length  Slot  Name   Signature
               0      10     0  this   Lcom/jvm/stack/Test1;
               0      10     1     a   I
               0      10     2     b   I
               4       6     3     c   I
         */
    }
}
  1. slot是复用的,以节省栈帧的空间,这种设计可能会影响到系统的垃圾收集行为。
// -Xms10m -Xmx10m
public static void main(String[] args) {
  {
      byte[] bs1 = new byte[1024 * 1204];
      byte[] bs2 = new byte[1024 * 1204];
      byte[] bs3 = new byte[1024 * 1204];

      /* 此时slot的情况
      0 -- args -- 堆引用
      1 -- bs1  -- 堆引用
      2 -- bs2  -- 堆引用
      3 -- bs3  -- 堆引用
      */

      System.gc();
      printMemory(); // freeMemory :2.98663330078125

      bs1 = null;
      /* 显式将bs1置为null, slot 1则空闲出来*/
  }
  System.gc();
  printMemory(); // freeMemory :4.8661346435546875

  /* 代码块结束,上述的bs1 bs2 bs3变量的作用域已经结束,
  所占用的slot可以被后续定义的变量复用 */

  int a = 5;
  int b = 5;

  /* 此时slot的使用情况
      0 -- args -- 堆引用
      1 -- a    -- I
      2 -- b    -- I
      3 -- bs3  -- 堆引用
   */

  System.gc();
  printMemory(); // freeMemory :6.8662872314453125

  String c = "a";
  /* 此时slot的使用情况
     0 -- args -- 堆引用
     1 -- a    -- I
     2 -- b    -- I
     3 -- c  -- 堆引用
   */

  System.gc();
  printMemory(); // freeMemory :8.866722106933594
}

public static void printMemory() {
  //System.out.println("totalMemory:" + Runtime.getRuntime().totalMemory()/1024.0/1024.0);
  System.out.println("freeMemory :" + Runtime.getRuntime().freeMemory()/1024.0/1024.0);
  //System.out.println("maxMemory  :" + Runtime.getRuntime().maxMemory()/1024.0/1024.0);
}

4) 操作数栈

  1. 操作数栈:用来存放方法运行期间,各个指令操作的数据。
  2. 操作数栈中元素的数据类型必须和字节码指令的顺序严格匹配
    • 数据类型和插槽位置的数据类型必须一一对应:比如指令 iconst_1,那么插槽1位置的类型必须是I
  3. 虚拟机在实现栈帧的时候可能会做一些优化,让两个栈帧出现部分重叠区域,以存放公用的数据
package com.lc.sprnigcloud.stack;

/**
 * @author hahadasheng
 * @since 2020/12/28
 */
public class Test {

    public static void main(String[] args) {
        test();
    }

    /* LocalVariableTable:
    Start  Length  Slot  Name   Signature
        2      51     0     a   I
       16      37     1     b   I
       35      18     2     c   I
     */
    public static void test() {
        int a = 1;
        a = a++;
        /*
         0: iconst_1            -> 常数1
         1: istore_0            -> 将常数1存放在局部变量表slot_0的位置,也就是a
         2: iload_0             -> 将slot_0中存放a的值1 放入栈中
         3: iinc          0, 1  -> slot_0中a的值自增1,1 + 1 = 2
         6: istore_0            -> 将栈中的值1赋值到slot_0的位置,所以此时a的值还是为1
         */
        System.out.println(a); // 1

        int b = 1;
        b = b++ * ++b;
        /*
        14: iconst_1            -> 常数1
        15: istore_1            -> 将1放在slot_1的位置
        16: iload_1             -> 将slot_1的值1入栈 栈:[1]
        17: iinc          1, 1  -> slot_1位置的1自增1,1+1=2
        20: iinc          1, 1  -> slot_1位置的2自增1,2+1=3
        23: iload_1             -> 将slot_1的值3入栈 栈:[3,1]
        24: imul                -> 将栈中的两个数相乘 3 * 1 = 3,栈中的值为3
        25: istore_1            -> 将栈中值3放在slot_1的位置
         */
        System.out.println(b); // 3

        int c = 1;
        c = ++c * c++;
        /*
        33: iconst_1            -> 常数1
        34: istore_2            -> 将常数1放在局部变量表slot_2的位置
        35: iinc          2, 1  -> slot_2位置数自增1,1+1=2
        38: iload_2             -> 将slot_2位置的2入栈 栈:[2]
        39: iload_2             -> 将slot_2位置的2入栈 栈:[2,2]
        40: iinc          2, 1  -> 将slot_2位置的2自增1,2+1=3
        43: imul                -> 将栈中的值取出来进行乘法操作 2*2=4,栈中的值变为4
        44: istore_2            -> 将栈中值4存放在slot_2的位置
         */
        System.out.println(c); // 4
    }
}

5) 动态连接

  1. 动态连接:每个栈帧持有一个指向运行时常量池中该栈帧所属方法的引用,以支持方法调用过程中的动态连接。
  2. 动态连接分类:
    1. 静态解析:类加载的时候,符号引用就转化成直接引用
    2. 动态连接:运行期间转换为直接引用(动态分派)

6) 方法返回地址

  1. 方法返回地址:方法执行后返回的地址,
    • 无论正常退出还是异常退出,都得返回到方法被调用的位置,程序才能继续执行

7) 方法调用

  1. 方法调用:方法调用就是确定具体调用哪一个方法,并不涉及方法内部的执行过程。
    • 不一定要调用这个方法
  2. 部分方法是直接在类加载的解析阶段,就确定了直接引用关系
    • 静态方法、私有方法、实例构造器、父类方法
  3. 但是对于实例方法,也称虚方法,因为重载和多态,需要运行期间动态委派
    • 例如虚拟机调用此方法的指令为 invokevirtual

8) 静态分派和动态分派

  1. 分派:分为静态分派和动态分派

  2. 静态分派:所有依赖静态类型来定位方法执行版本的分派方式

    • 比如:重载方法(依据传参确定)
  3. 动态分派:根据运行期间的实际类型来定位方法执行版本的分派方式,

    • 比如:重写覆盖方法、实现的接口等
  4. 单分派和多分派:就是按照分派思考的纬度,多于一个的就算多分派,只有一个的称为单分派

    • 只有一个确认的,没有重载、重写、多态等可能有多个可能的就是单分派
  5. 如何执行方法中的字节码指令:JVM通过基于栈的字节码解释器引擎来执行指令,JVM的指令集也是基于栈的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容