JVM技术之旅-GCRoots定位及Mutator线程中断

简介: JVM技术之旅-GCRoots定位及Mutator线程中断

前提概要


大多数的垃圾收集器在在收集垃圾的时候会停顿所有的线程(Stop The World)来进行可达性分析,那么如何快速找到GC Roots?线程应该在什么地方停止呢?

image.png

快速找到GC ROOTS:OopMap


当所有线程停下来的时候,并不需要一个不漏的检查完所有执行上下文和全局引用位置,虚拟机应该是有办法直接知道哪些地方存放着对象引用。在HotSpot的实现中,是使用一组称为OopMap的数据结构来达到目的的




OopMap存储两种对象引用


对象内的引用


类加载完的时候,HotSpot就把对象内什么偏移量上是什么类型的数据计算出来。



栈和寄存器引用


在JIT编辑过程中,也会在特定的位置记录下栈和寄存器中哪些位置是引用。这样,GC在扫描的时候就知道这些信息了。通过OopMap垃圾收集器就可以更快的找到GC Roots,并且更快的完成GC

Roots的枚举,大概展示如下图:


image.png




线程中断点:安全点(Safe Point)


定义


安全点就是程序能够停顿的位置。即程序不是在任何时候停顿下来进行GC,只有到了安全点才去更新OopMap和停顿,等待GC完成在继续执行。



分析


OopMap,HotSpot就能很快的完成GC Roots的枚举了。但是问题来了,每一行代码都有可能使引用变化,就需要更新OopMap,在哪个位置去更新OopMap呢?如果每一行都执行一次更新,肯定是不科学的,所以就有了安全点(safe point)。


安全点设置太多肯定不行,造成运行压力,太少的话两个点之间太长,如果刚过第一个安全点然后要求GC,但是程序要运行到下一个安全点才能停下来,那么GC等待的时间就太长了。



安全点选择标准


是否就有让程序长时间执行的特征。一条指令执行时间都很短,而一段程序一般不会说因为很长的指令流而造成长时间的运行,所以一般都是在指令复用的地方出现。比如:方法调用、循环跳转、异常跳转





让线程停下来的两种方法


抢断式中断


GC发生时,中断直接所有线程,发现没有在安全点的,再恢复线程让他跑到安全点。现在几乎没有虚拟机采用这种方式。



主动式中断


当GC需要中断线程时,设置一个标志,各个线程去轮询这个标志,发现需要中断,线程就自己中断。轮询点和安全点在一个地方,在加上创建对象需要分配内存的地方。



实现方式:


设置一个内存块不可读,当线程访问这个内存就会产生一个自陷异常信号,预先注册的异常处理器中捕获这个异常暂停线程。通过一个指令和一个异常处理器就实现了这个功能





休眠线程如何中断:安全区域(Safe Region)


安全点不能解决的问题


安全点解决了正在执行的线程中断问题,我们知道线程还有没执行的状态,比如线程是Sleep、Blocked状态。这些线程不能自己走到安全点。如果休眠的线程在GC途中醒来,在线程运行到安全点之前就会有可能修改对象的引用关系。所以我们需要在线程醒来的时候如果正在GC那么也中断。



安全区域


安全区域就是在一段代码中引用关系不会发生变化。所以在这个区域内任何地方GC都是安全的。在执行到安全区的时线程会标识自己处于安全区中,当离开安全区时,就需要检查系统是否已经完成执行GC Roots(或者整个GC过程),如果已经完成那么线程继续执行,否则就等待



安全区域举例


线程的Sleep、Blocked(这个区域内当前线程肯定不会改变对象引用)就被包含在安全区中,也就是说只要线程Sleep那么他就处于安全区,一旦Sleep时间到线程继续执行,首先就要判断是否能够离开安全区。





GC基础全面总结


可以猜测GC大概流程


线程运行时创建OopMap,当需要GC时,所以执行中的线程跑到安全点然后中断,不在执行中的线程处在安全安全区域。此时遍历OopMap(相当于GC Roots)根据可达性分析,标记出存活对象,然后根据区域和具体实现进行清除算法、复制算法、整理算法。GC完成线程继续执行,如果在GC过程中从Sleep醒来的线程且允许离开安全区之后也可以继续执行。


image.png


本章总结:


  • 虚拟机中实现所有线程到达安全点自动中断功能只用了一个标志位和一个异常处理器。为我们有时候遇到一些全局功能时提供了一种解决思路


  • 既然正在休眠类的线程不能走到安全点,那就把线程休眠的周围设置成一块区域,离开区域的时候先问问能不能离开





相关文章
|
26天前
|
安全 Java 调度
Java语言多线程编程技术深度解析
Java语言多线程编程技术深度解析
291 1
|
2天前
|
存储 Java C++
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据
Java虚拟机(JVM)管理内存划分为多个区域:程序计数器记录线程执行位置;虚拟机栈存储线程私有数据,如局部变量和操作数;本地方法栈支持native方法;堆存放所有线程的对象实例,由垃圾回收管理;方法区(在Java 8后变为元空间)存储类信息和常量;运行时常量池是方法区一部分,保存符号引用和常量;直接内存非JVM规范定义,手动管理,通过Buffer类使用。Java 8后,永久代被元空间取代,G1成为默认GC。
10 2
|
7天前
|
Java
【技术瑜伽师】Java 线程:修炼生命周期的平衡之道,达到多线程编程的最高境界!
【6月更文挑战第19天】Java多线程编程犹如瑜伽修行,从创建线程开始,如`new Thread(Runnable)`,到启动线程的活跃,用`start()`赋予生命。面对竞争与冲突,借助同步机制保证资源访问的有序,如`synchronized`关键字。线程可能阻塞等待,如同瑜伽的静止与耐心。完成任务后线程终止,整个过程需密切关注状态变换,以求多线程间的和谐与平衡。持续修炼,如同瑜伽般持之以恒,实现高效稳定的多线程程序。
|
7天前
|
Java 开发者
【技术成长日记】Java 线程的自我修养:从新手到大师的生命周期修炼手册!
【6月更文挑战第19天】Java线程之旅,从新手到大师的进阶之路:始于创建线程的懵懂,理解就绪与运行状态的成长,克服同步难题的进阶,至洞悉生命周期的精通。通过实例,展示线程的创建、运行与同步,展现技能的不断提升与升华。
|
7天前
|
Java
【技术解码】Java线程的五味人生:新建、就绪、运行、阻塞与死亡的哲学解读!
【6月更文挑战第19天】Java线程生命周期如同人生旅程,经历新建、就绪、运行、阻塞至死亡五阶段。从`new Thread()`的诞生到`start()`的蓄势待发,再到`run()`的全力以赴,线程在代码中奔跑。阻塞时面临挑战,等待资源释放,最终通过`join()`或中断结束生命。线程的每个状态转变,都是编程世界与哲思的交汇点。
|
1天前
|
存储 安全 Java
Java中的线程安全与同步技术
Java中的线程安全与同步技术
|
26天前
|
安全 Java
JAVA语言中的多线程编程技术
JAVA语言中的多线程编程技术
|
26天前
|
存储 安全 Java
Java语言中的多线程编程技术深入解析
Java语言中的多线程编程技术深入解析
221 1
|
1月前
|
Java 调度
Java中线程池的技术
Java中线程池的技术
31 3
|
1月前
|
Dart 前端开发 安全
【Flutter前端技术开发专栏】Flutter中的线程与并发编程实践
【4月更文挑战第30天】本文探讨了Flutter中线程管理和并发编程的关键性,强调其对应用性能和用户体验的影响。Dart语言提供了`async`、`await`、`Stream`和`Future`等原生异步支持。Flutter采用事件驱动的单线程模型,通过`Isolate`实现线程隔离。实践中,可利用`async/await`、`StreamBuilder`和`Isolate`处理异步任务,同时注意线程安全和性能调优。参考文献包括Dart异步编程、Flutter线程模型和DevTools文档。
【Flutter前端技术开发专栏】Flutter中的线程与并发编程实践