栈帧中都有啥东西

栏目: Java · 发布时间: 5年前

内容简介:我的所有文章同步更新与Github--栈帧是虚拟机栈的一个单位,之前讲解了运行时数据区和类加载的过程,现在我们看下虚拟机中栈帧都是啥样子的,这个应该算是运行时数据区(JVM内存结构的补充),如果不了解可以参考我的这篇博文运行时栈帧中存储了以下内容

我的所有文章同步更新与Github-- Java-Notes ,想了解JVM,HashMap源码分析,spring相关,剑指offer题解(Java版),可以点个star。可以看我的github主页,每天都在更新哟。

邀请您跟我一同完成 repo

栈帧是虚拟机栈的一个单位,之前讲解了运行时数据区和类加载的过程,现在我们看下虚拟机中栈帧都是啥样子的,这个应该算是运行时数据区(JVM内存结构的补充),如果不了解可以参考我的这篇博文 JVM内存结构

运行时栈帧结构

运行时栈帧中存储了以下内容

  • 局部变量
  • 操作数栈
  • 动态链接
  • 返回地址
  • 附加信息
  • ….
栈帧中都有啥东西

每一个方法的调用开始和结束都是栈的压入(入栈)和弹出(出栈)的过程

局部变量表

是什么

局部变量表是 一组变量值存储空间 ,用于存放 方法参数方法内部 定义的 局部变量

大小是编译的时候写进了字节码里面的,在Code属性中的max_local属性,即下面的local

栈帧中都有啥东西

有什么

局部变量表的容量以 变量槽 (Variable Slot)为 最小单位 ,虚拟机中并没有明确指明一个Slot应占用的内存空间大小,只是很有导向性的说到每个Slot都 应该能存放 一个下面8种类型的其中一个。(如果是long或者double这种64位的数据类型,则需要两个Slot)

  • boolean
  • byte
  • char
  • short
  • int
  • float
  • reference
  • returnAddress

前六种应该不用说,是基本的数据类型,reference是啥呢

reference

reference是一个对象实例的引用

作用:

  • 从此引用中 直接或间接查找对象在 Java 中的数据存放的 起始地址索引
  • 从此引用中 直接或间接查找 到对象所属数据类型在 方法区 中的 存储的类型对象 (因为类信息在方法区中存储)

returnAddress

为字节码指令jsr、jsr_w和ret提供的,指向了一条字节码指令的地址

已经很少见了。

注意

局部变量表中的 局部变量 和之前将类加载的时候的 类变量 (static修饰)不一样,他没有所谓的"准备阶段",所以没有设置初始值的阶段。不知道的可以参考我类加载这篇文章,看了准备阶段,应该就知道了。 类加载过程

所以我们在写程序的时候这样写,对比你就知道了

栈帧中都有啥东西

其他类型零值

栈帧中都有啥东西

Slot复用

不使用的对象,应当手动赋值为null

为了尽可能节省栈空间,局部变量表的 Slot可以复用 。方法体中定义的变量, 其作用域并不一定覆盖整个方法体 ,如果当前字节码PC计数器的值已经超出了某个变量的作用域,那这个变量对应的Slot就可以交给其他变量使用。

不过这样的做法,会有一些缺点,我们来看下面的代码示例

public class Test2 {

    public static void main(String[] args) {
        

            byte[] placeholder = new byte[64 * 1024 * 1024];


        
        System.gc();
    }
}
复制代码

我们通过配置虚拟机参数 -verbose:gc 来打印垃圾回收的结果

栈帧中都有啥东西

我们看到他并没有回收。

我们修改一下代码

public class Test2 {

    public static void main(String[] args) {
        {

            byte[] placeholder = new byte[64 * 1024 * 1024];


        }
        System.gc();
    }
}
复制代码
栈帧中都有啥东西

他还是没有进行回收,按理说 placeHolder 的作用域只在花括号中,在执行gc方法的时候,他就已经不可能用了,算是已经"死"了的对象了,为什么没有回收呢?

我们再修改一下

public class Test2 {

    public static void main(String[] args) {
        {

            byte[] placeholder = new byte[64 * 1024 * 1024];


        }
      	int  a = 0;
        System.gc();
    }
}
复制代码
栈帧中都有啥东西

我们看到,这次垃圾回收器工作了,为什么呢?

placeholder 能否被回收的根本原因是: 局部变量表的Slot是否还存有关于placeholder数组对象的引用。

第一次修改中,代码虽然离开了该变量的作用域,但是在此之后, 没有任何对局部变量表的读写操作 ,该变量 原本占用的Slot 还没有被任何其他变量 复用 ,所以作为GC Root 一部分的局部变量表仍然保持着对他的关联(不了解什么可以作为GC Root的话,可以参考我的这篇文章 JVM垃圾回收 )

而第二次,则改变了上面的这种情况

所以当遇到一个方法,其后面的代码有一些耗时很长的操作,而前面又定义了占用大量内存、实际上已经用不到的变量,应当手动设置其为null。

很多 工具 类都有这个操作,比如 ArrayList和Stack中的remove方法,你也可以找下其他的工具类中的方法,看是否有此类操作

栈帧中都有啥东西
栈帧中都有啥东西

操作数栈

操作数栈记录了一个方法执行过程中的字节码指令,他往操作数栈中进行入栈和出栈

大小在编译的时候也已经确定了,字节码文件中的Code属性中的max_stacks数据项,即下图的 stack

栈帧中都有啥东西

当一个方法刚执行的时候,操作数栈是空的,在方法执行的过程中,会有各种字节码指令往操作数栈中入栈和出栈。

动态连接

每一个栈帧都包含一个 指向运行时常量池 中该栈帧所属的 方法的引用 ,持有这个引用是为了支持方法调用过程中的 动态连接

如果你看了 字节码文件构成类加载过程 ,你应该知道,字节码文件中有很多符号引用。

这些符号引用一部分会在 类加载的解析阶段 或者 第一次使用的时候 转化为直接引用,这种转化称为 静态解析

另一部分会在 运行期间 转化为直接引用,这部分称为 动态连接

动态连接会在这篇文章 方法调用 中讲解


以上所述就是小编给大家介绍的《栈帧中都有啥东西》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

浅薄

浅薄

[美]尼古拉斯·卡尔 / 刘纯毅 / 中信出版社 / 2015-11 / 49.00 元

互联网时代的飞速发展带来了各行各业效率的提升和生活的便利,但卡尔指出,当我们每天在翻看手机上的社交平台,阅读那些看似有趣和有深度的文章时,在我们尽情享受互联网慷慨施舍的过程中,我们正在渐渐丧失深度阅读和深度思考的能力。 互联网鼓励我们蜻蜓点水般地从多种信息来源中广泛采集碎片化的信息,其伦理规范就是工业主义,这是一套速度至上、效率至上的伦理,也是一套产量最优化、消费最优化的伦理——如此说来,互......一起来看看 《浅薄》 这本书的介绍吧!

MD5 加密
MD5 加密

MD5 加密工具

SHA 加密
SHA 加密

SHA 加密工具

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具