0

0

C# 垃圾回收(GC)机制是如何工作的 - 深入理解.NET内存管理

星降

星降

发布时间:2025-12-05 14:02:02

|

466人浏览过

|

来源于php中文网

原创

C#垃圾回收是分代、可干预的内存管理机制,按0/1/2代划分对象生命周期,通过根引用链判定存活,支持低延迟模式、NOGC区域等配置优化。

c# 垃圾回收(gc)机制是如何工作的 - 深入理解.net内存管理

C# 的垃圾回收(GC)不是“自动清理内存”的黑箱,而是一套有策略、分代、可干预的内存管理机制。 它在后台持续监控对象生命周期,按需回收不可达对象所占堆内存,但它的行为受代码写法、对象大小、代际划分和运行时环境共同影响——理解这些细节,才能真正避免内存泄漏和 GC 频繁暂停。

堆内存被划分为三代:0代、1代、2代

.NET GC 采用“分代回收”策略,核心假设是:大部分对象生命周期很短,少数长期存活的对象应减少扫描频率。因此托管堆被逻辑划分为三代:

  • 0代:最新分配的对象所在区域,GC 最频繁触发(如局部变量、临时集合)。回收后仍存活的对象会被提升到 1 代;
  • 1代:中等生命周期对象,GC 触发频率较低,主要作为 0→2 的缓冲;
  • 2代:长期存活对象(如静态缓存、全局服务实例),GC 开销最大,只在必要时(如内存压力大或调用 GC.Collect(2))才回收。

代际提升不是复制移动就是标记压缩——小对象堆(SOH)在回收后会整理内存以减少碎片;大对象堆(LOH,≥85,000 字节)不整理,只做标记清除,容易产生内存碎片。

GC 触发条件不只是“内存不够”

触发 GC 并非等到物理内存耗尽。常见触发场景包括:

  • 0代内存分配超过阈值(由运行时动态估算,与历史分配速率相关);
  • 显式调用 GC.Collect()(不推荐,干扰运行时优化);
  • 系统内存不足通知(Windows 的 MEMORYSTATUS 或 Linux 的 cgroup 压力信号);
  • 应用程序空闲且后台 GC 线程启动(.NET Core 3.0+ 的后台 GC 模式)。

注意:GC.Collect() 默认只回收 0 代;强制全代回收会阻塞当前线程,并可能引发更长暂停——它应是诊断手段,而非常规优化方式。

Anyword
Anyword

AI文案写作助手和文本生成器,具有可预测结果的文案 AI

下载

对象“存活”与否取决于根引用链是否可达

GC 判定对象是否可回收,本质是图遍历:从所有“根”(Roots)出发,沿着引用关系向下搜索,所有能到达的对象视为“存活”,其余标记为垃圾。

  • 根包括:全局/静态变量、正在执行的方法的局部变量和参数、寄存器中的引用、线程上的引用、Finalizer 队列中的对象;
  • 常见“隐式根”陷阱:事件订阅未取消(obj.Event += Handler 使 obj 被 EventHandler 持有)、静态集合 Add 了实例对象、Timer/Task 回调持有 this 引用;
  • 弱引用(WeakReference) 可打破强引用链,让对象在 GC 时能被回收,适合缓存场景。

可控的 GC 行为:配置与提示

虽然不能完全控制 GC 时间点,但可通过以下方式引导其行为:

  • 使用 GC.TryStartNoGCRegion(sizeInBytes) 在关键路径(如实时音频处理)临时禁止 GC,确保低延迟;
  • 设置 GC 模式:GCSettings.LatencyMode = GCLatencyMode.LowLatency(短期启用,需及时恢复);
  • 大对象尽量复用或使用 ArrayPool / MemoryPool 减少 LOH 分配;
  • 实现 IDisposable 并在 Dispose() 中释放非托管资源(文件句柄、数据库连接等),避免依赖 Finalizer——Finalizer 执行时机不确定,且会延长对象生命周期(至少多一次 GC)。

基本上就这些。GC 不是魔法,它高效的前提是你写的代码尊重它的规则:减少不必要的长引用、及时解耦、合理使用池和弱引用。真正影响性能的往往不是 GC 本身,而是我们无意中制造的“该死的存活对象”。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

621

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1104

2023.07.27

windows照片无法显示
windows照片无法显示

当我们尝试打开一张图片时,可能会出现一个错误提示,提示说"Windows照片查看器无法显示此图片,因为计算机上的可用内存不足",本专题为大家提供windows照片无法显示相关的文章,帮助大家解决该问题。

792

2023.08.01

云朵浏览器入口合集
云朵浏览器入口合集

本专题整合了云朵浏览器入口合集,阅读专题下面的文章了解更多详细地址。

20

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
php-src源码分析探索
php-src源码分析探索

共6课时 | 0.5万人学习

Golang云原生架构师课程
Golang云原生架构师课程

共49课时 | 3.1万人学习

Golang基础入门到精通(第二季)
Golang基础入门到精通(第二季)

共49课时 | 2.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号