对jvm中 GC Roots 枚举的理解

对 GC Roots 枚举的理解

这里,只是在理论上讲,全堆垃圾收集。不包含现实中垃圾收集器的具体实现,也就是不包含分代、分区垃圾收集,其实,也就多一个跨代引用、跨区引用的问题。

啥是 GC Roots ?

凡是有堆外引用的对象,都是 GC Root。

局部变量(栈里)、静态变量和常量(方法区里),它们都是堆外的,只要引用了对象,那这个对象就是 GC Root ,就会被加入GC Roots 集合。


设想,这个世界上还没有垃圾收集器,而我想要发明一个,那遇到的第一个问题就是,我该如何判断一个对象是否已经死了。有一个方法,就是从当前肯定活着的对象入手。如果,我能把所有活着的对象找齐了,那剩下的就是死的了,逻辑上就是这么简单。对于活对象,我更愿意称它们为有用的对象,因为这更能表达它不应该被清理的特征。

只要两个条件,就可以找到jvm里所有,有用的对象:

1、栈、方法区等,非堆的内存区域,持有的对象。

堆里存了所有的对象,它们就像仓库里的工具,只有被使用才是有用的。被哪里使用呢?被堆外的其他区域。


2、第一个条件里的对象,在堆内使用的所有对象。

使用的意思是,它调用的对象,所调用的对象,所调用的对象...(套娃中)。是一个调用链。


GC Roots 就是为了实现第一个条件,它要找到所有被堆外区域引用的对象。

可达性分析,就是为了实现第二个条件,它要找到 GC Roots 所使用的所有对象。



GC Roots 枚举的过程

stop the world

GC Roots 枚举,必然需要暂停所有用户线程。

原因很好理解:GC Roots 枚举时,统计的就是 堆外 引用的对象。只拿栈中来说,如果客户线程依旧在运行,那么统计过程中,不断有栈帧出栈和入栈。新入栈的栈帧,其中的局部变量,也有可能是不会被统计到 GC Roots 里。(看完OopMap的机制,再更新详细内容)



safe point

在任何时候,都可以暂停用户线程。为啥要设置 safe point ?

理解这个问题,可能需要先搞懂 OopMap,但是,我的理解暂时还不透彻,所以只能笼统的讲。

用户线程暂停后,接下来会进行 GC Roots 枚举,而 GC Roots 枚举,却依赖 OopMap 存储的信息。但是,OppMap 不能每时每刻都更新数据,因为那样的话,对资源的消耗就太大了。

所以,要给 OopMap 更新数据,找一个节点,这个节点就是 safe point 。

在选取 safe point 时,只会选择在循环调用递归 。。。这是为什么呢?

JVM暂停用户线程的时候,都是采用 主动式暂停 ,所以以下就用 主动式暂停 来讨论。当 "stop the world"发生时,线程应该尽快跑到安全点,然后停下来。如果在选取安全点的时候,选在了 循环语句 的后面,那就太蠢了,白白增加了停顿时间。所以,肯定应该选在 循环语句 运行之前,基本就是在这个 循环语句 的地方。

总结,就是安全点的选择,会在长时间运行的地方,比如 循环调用 等地方。并在这些代码运行之前,暂停线程,避免浪费时间。



safe region

safe region 是 safe point 的补丁。

如果,在 "stop the world" 发生时,有些用户线程没有运行,比如 sleep 了。那么,这条线程, 在 "stop the world" 发生时,它不是运行状态,所以它并不能自行跑到安全点,并中断线程。而是,一直停在了非安全点的位置,一旦获取了时间片,它还是要运行的。

可能会在,下一个致命的时期,也就是 GC Roots 枚举时,这条线程如果又跑起来了。

那这个漏洞就太大了。

所以,又引入了 safe region,来补这个漏洞。


当用户线程执行到安全区域里面的代码时,首先会标识自己已经进入了安全区域,那样当这段时间里虚拟机要发起垃圾收集时就不必去管这些已声明自己在安区域内的线程了。

当线程要离开安全区域时,它要检查虚拟机是否已经完成了根节点枚举(或者垃圾收集过程中其他需要暂停用户线程的阶段),如果完成了,那线程就当作没事发生过,继续执行;否则它就必须一直等待,直到收到可以离开安全区域的信号为止。

——摘抄自《深入理解java虚拟机》第三版



总结

GC Roots 枚举前。首先,要暂停用户线程,会使用 safe point 和 safe ragion ,辅助暂停用户线程。然后,开始 GC Roots 枚举,并不会去遍历方法区、栈里的引用,而是,直接从OopMap里获取引用。

java程序员,linux系统爱好者。

44 声望
3 粉丝
0 条评论
推荐阅读
Spring事务传播行为详解
Spring在TransactionDefinition接口中规定了7种类型的事务传播行为。事务传播行为是Spring框架独有的事务增强特性,他不属于的事务实际提供方数据库行为。这是Spring为我们提供的强大的工具箱,使用事务传播行可...

JerryTse242阅读 122.6k评论 97

一文搞懂秒杀系统,欢迎参与开源,提交PR,提高竞争力。早日上岸,升职加薪。
前言秒杀和高并发是面试的高频考点,也是我们做电商项目必知必会的场景。欢迎大家参与我们的开源项目,提交PR,提高竞争力。早日上岸,升职加薪。知识点详解秒杀系统架构图秒杀流程图秒杀系统设计这篇文章一万多...

王中阳Go31阅读 2.3k评论 1

封面图
计算机网络连环炮40问
本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点,欢迎star~

程序员大彬13阅读 1.7k

万字详解,吃透 MongoDB!
MongoDB 是一个基于 分布式文件存储 的开源 NoSQL 数据库系统,由 C++ 编写的。MongoDB 提供了 面向文档 的存储方式,操作起来比较简单和容易,支持“无模式”的数据建模,可以存储比较复杂的数据类型,是一款非常...

JavaGuide8阅读 1.6k

封面图
花了半个小时基于 ChatGPT 搭建了一个微信机器人
相信大家最近被 ChatGPT 刷屏了,其实在差不多一个月前就火过一次,不会那会好像只在程序员的圈子里面火起来了,并没有被大众认知到,不知道最近是因为什么又火起来了,而且这次搞的人尽皆知。

Java极客技术12阅读 3k评论 3

封面图
数据结构与算法:二分查找
一、常见数据结构简单数据结构(必须理解和掌握)有序数据结构:栈、队列、链表。有序数据结构省空间(储存空间小)无序数据结构:集合、字典、散列表,无序数据结构省时间(读取时间快)复杂数据结构树、 堆图二...

白鲸鱼9阅读 5.2k

PHP转Go实践:xjson解析神器「开源工具集」
我和劲仔都是PHP转Go,身边越来越多做PHP的朋友也逐渐在用Go进行重构,重构过程中,会发现php的json解析操作(系列化与反序列化)是真的香,弱类型语言的各种隐式类型转换,很大程度的减低了程序的复杂度。

王中阳Go11阅读 2.6k评论 4

封面图

java程序员,linux系统爱好者。

44 声望
3 粉丝
宣传栏