【GC的过程】

简介: 【GC的过程】

GC(垃圾回收)是程序自动管理内存的一种机制,通过扫描内存中的对象和引用,自动确定哪些对象是已死的(即无法再被访问),并将这些对象标记为垃圾,释放它们所占用的内存空间。下面我们将介绍GC的过程,并附上Java代码示例来说明。


1. 标记阶段

在标记阶段,GC扫描所有的对象,找出所有可达对象,这些对象是指在应用程序中仍然被引用的对象,将这些对象标记为存活对象。未被标记的对象就会被判定为垃圾。为了实现标记过程,GC需要对整个堆进行扫描,记录哪些对象是存活对象。

Java代码示例:

public class GCExample {
   public static void main(String[] args) {
      // 创建对象
      Object obj1 = new Object();
      Object obj2 = new Object();
      Object obj3 = new Object();
      // 将obj1和obj2设为相互引用
      obj1.setReference(obj2);
      obj2.setReference(obj1);
      // 将obj3设为root对象
      root = obj3;
      // 执行标记过程
      mark(root);
   }
   // 在标记过程中,使用递归算法遍历所有存活对象
   public static void mark(Object obj) {
      if (obj.isMarked()) {
         return;
      }
      obj.mark();
      Object[] references = obj.getReferences();
      for (Object ref : references) {
         mark(ref);
      }
   }
}

在上面的代码示例中,我们创建了三个对象obj1、obj2和obj3,并将obj1和obj2相互引用,将obj3设置为root对象。在执行mark方法时,从根对象obj3开始,使用递归算法遍历所有存活对象,将遍历到的对象标记为存活对象。

2. 清除阶段

在清除阶段,GC清除被标记为垃圾的对象,收回它们所占用的内存空间。清除过程是在后台进行的,应用程序不会受到影响。在清除过程中,GC会将被标记为垃圾的对象从堆中清除。这些对象的内存空间将被释放,以便可供后续程序使用。

Java代码示例:

public static void sweep() {
   for (Object obj : heap) {
      if (!obj.isMarked()) {
         heap.remove(obj);
      } else {
         obj.unmark();
      }
   }
}

在上面的代码示例中,我们遍历整个堆,对于所有没有被标记的对象,将其从堆中清除。而对于被标记了的对象,将其标记状态重置为未标记。

3. 压缩阶段

在压缩阶段,GC会将存活对象向堆的一端移动,以便获得一块连续的内存空间。这个过程也称为内存整理。这个过程的目的是使内存空间更加紧凑,并且减少内存碎片的数量,从而提高内存利用率。

Java代码示例:

public static void compact() {
   int offset = 0;
   for (Object obj : heap) {
      if (obj.isMarked()) {
         obj.move(offset);
         offset += obj.getSize();
      }
   }
}

在上面的代码示例中,我们遍历所有存活对象,将它们向堆的一端移动,并更新它们在堆中的位置。在移动对象时,我们需要记录它们的偏移量,以便正确地更新它们的引用。

以上就是GC的三个阶段:标记阶段、清除阶段和压缩阶段。这些阶段的实现方式因开发语言而异,但是它们的基本原理是相同的。

小故事

有一位名叫小明的程序员,在工作中经常使用Java语言编写代码。他知道Java的优点之一是可以自动进行垃圾回收,也就是GC。但是他对GC的具体过程还不是很了解。有一天,他向同事请教,同事给他讲了一个小故事来说明GC的过程:

故事是这样的:有一位园丁在花园里种了一些花,他会定期巡视花园,将枯萎的花朵摘掉。这样,花园里的花就能保持鲜艳美丽。这位园丁就像是JVM的垃圾回收器,花就像是程序中的不再使用的对象。

在Java程序中,当一个对象不再被引用时,垃圾回收器就会将其标记为垃圾对象,并将其从内存中清除。这个过程分为两部分:标记和清除。

标记阶段:垃圾回收器会遍历堆中的所有对象,并标记哪些对象是可达的(被引用的)和哪些对象是不可达的(未被引用的)。

清除阶段:垃圾回收器会清除所有的不可达对象,并释放它们占用的内存空间。

就像园丁巡视花园一样,垃圾回收器需要周期性地运行,以便及时地清理垃圾对象,释放内存。如果垃圾回收器不能及时地清除垃圾对象,程序可能会因为内存泄漏而出现问题,甚至导致程序崩溃。

通过这个小故事,小明理解了JVM GC的过程,他感到更加安心地使用Java语言编写代码。


相关文章
|
8月前
|
缓存 算法 Java
GC 的三种基本实现方式
GC 的三种基本实现方式
85 1
|
4月前
|
存储 Java
jvm性能调优实战 - 23 模拟Young GC的发生及分析GC日志
jvm性能调优实战 - 23 模拟Young GC的发生及分析GC日志
45 0
|
10月前
|
存储 算法 Java
jvm之垃圾回收清除算法解读
jvm之垃圾回收清除算法解读
|
10月前
|
数据可视化 Java 应用服务中间件
GC 分析
GC 分析
75 0
|
11月前
|
Java
JVM GC频繁优化
JVM GC频繁优化
165 0
|
12月前
|
算法 Java
JVM的GC机制和常见GC算法
@[toc] 1. 堆内存的分代 2. GC分类 3. 什么是GC 3.1 需要GC的内存区域 3.2 GC回收的对象 3.3 判断对象存活的两种算法 3.3.1 引用计数 3.3.2 可达性分析 3.4 什么时候触发GC 4. 常见的GC算法 4.1 标记-清除算法 4.2 复制算法 4.3 标记-压缩算法
96 0
|
12月前
|
存储 SQL 算法
彻底理解对象内存分配及Minor GC和Full GC全过程
某数据计算系统,日处理亿级数据量。系统不断从各种数据源提读数据,加载到JVM内存进行计算处理
190 0
|
存储 缓存 算法
详述JVM的GC及垃圾回收策略
详述JVM的GC及垃圾回收策略
873 2
详述JVM的GC及垃圾回收策略
|
缓存 算法 Java
详解gc(垃圾回收)机制三:GC复制算法
详解gc(垃圾回收)机制三:GC复制算法
378 1
详解gc(垃圾回收)机制三:GC复制算法
|
算法 Java
详解gc(垃圾回收)机制四:GC标记-清除算法
详解gc(垃圾回收)机制四:GC标记-清除算法
102 1
详解gc(垃圾回收)机制四:GC标记-清除算法