0

0

C# 终结器队列Finalizer Queue C#对象析构和GC的Finalization过程

幻夢星雲

幻夢星雲

发布时间:2026-02-02 10:40:59

|

416人浏览过

|

来源于php中文网

原创

Finalizer Queue 是 GC 维护的带标记对象链表,非传统队列;对象在首次 GC 不可达且含终结器时加入,导致至少延迟一个 GC 周期回收,易因 Finalizer 线程阻塞或异常引发内存泄漏。

c# 终结器队列finalizer queue c#对象析构和gc的finalization过程

Finalizer Queue 是什么,它真在“排队”吗

Finalizer Queue 不是传统意义的队列,而是 GC 内部维护的一个**带标记的对象链表**,只存那些重写了 Finalize() 方法(或用 ~ClassName() 语法)且尚未被回收的实例。GC 不会按“先来后到”执行终结器,而是由 Finalizer 线程批量取出、逐个调用 Finalize() —— 所以“排队”只是逻辑概念,实际结构是链表 + 标记位。

关键点:只要对象有终结器,且没被 GC.SuppressFinalize() 显式抑制,就会被 GC 在第一次标记为不可达时加入该链表。

  • 对象进入 Finalizer Queue 的时机是:GC 第一次发现它不可达,且类型定义了终结器
  • 它不会出现在 Gen 0/1/2 的常规代际堆中,但它的引用关系会影响对象的代际晋升
  • Finalizer Queue 本身不占用大量内存,但它会让对象多活至少一个 GC 周期(甚至更久)

为什么重写 ~MyClass() 后对象迟迟不被回收

因为终结器让对象经历了“两次 GC 才能释放”的过程:第一次 GC 将其移入 Finalizer Queue 并标记为“待终结”,第二次 GC(或之后)才真正回收——前提是 Finalizer 线程已执行完 Finalize() 且对象不再被任何根引用。

常见诱因:

  • Finalizer 线程被阻塞(比如 Thread.Sleep()、锁竞争、同步 I/O)→ 整个队列卡住
  • Finalize() 中抛出未捕获异常 → 当前线程终止,后续终结器可能被跳过(.NET 5+ 默认终止进程)
  • 终结器里又创建了新对象并持有长生命周期引用 → 意外延长其他对象寿命
  • 高频分配带终结器的对象(如每帧 new 一个)→ Finalizer Queue 积压,GC 压力陡增

GC.SuppressFinalize(this) 应该在哪儿调用

必须在你**显式释放了非托管资源之后、且确定不再需要自动终结逻辑时**立即调用。典型场景是实现了 IDisposable 的类型,在 Dispose(bool disposing)disposing == true 分支末尾调用。

错误做法:

Morisot
Morisot

免提示词AI图片生成器,10秒生成十几张

下载
  • Finalize() 里调用 —— 此时已无意义,对象正被终结
  • 在构造函数失败时调用 —— 对象还没完全构建,this 可能无效
  • 仅在 Dispose() 外层调用,却忘了在 Dispose(true) 中调用 —— 导致 Finalize() 仍会被执行

正确模式:

public void Dispose()
{
    Dispose(true);
    GC.SuppressFinalize(this); // ← 这行必须有,且只在这里写一次
}

protected virtual void Dispose(bool disposing) { if (!_disposed) { if (disposing) { // 释放托管资源 _stream?.Dispose(); } // 释放非托管资源(如句柄、内存指针) if (_handle != IntPtr.Zero) { NativeMethods.CloseHandle(_handle); _handle = IntPtr.Zero; } _disposed = true; } }

Finalization 的真实开销和替代方案

终结器不是免费的:每个带终结器的对象会额外占用约 24 字节(.NET 6+)用于跟踪信息;Finalizer 线程是单线程,高负载下成为瓶颈;而且无法预测执行时机,不适合做超时控制、资源及时归还等场景。

现代 C# 更推荐:

  • 优先用 IDisposable + using 语句显式释放
  • 非托管资源封装用 SafeHandle 子类(它自带可靠的终结逻辑,且支持 Dispose() 抑制)
  • 需要异步清理时,用 IAsyncDisposable 而非依赖终结器
  • 完全避免重写 Finalize(),除非你真的在写类似 SafeHandle 的底层封装

GC 对终结器的处理机制本身很稳定,但滥用它带来的延迟、不可控性和调试难度,远超初学者预期。真正难的不是“怎么写终结器”,而是“怎么证明它根本不需要存在”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

399

2023.07.18

堆和栈区别
堆和栈区别

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

576

2023.08.10

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

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

546

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

87

2025.12.01

Java JNI 与本地代码交互实战
Java JNI 与本地代码交互实战

本专题系统讲解 Java 通过 JNI 调用 C/C++ 本地代码的核心机制,涵盖 JNI 基本原理、数据类型映射、内存管理、异常处理、性能优化策略以及典型应用场景(如高性能计算、底层库封装)。通过实战示例,帮助开发者掌握 Java 与本地代码混合开发的完整流程。

0

2026.02.02

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

61

2026.01.31

go语言 math包
go语言 math包

本专题整合了go语言math包相关内容,阅读专题下面的文章了解更多详细内容。

52

2026.01.31

go语言输入函数
go语言输入函数

本专题整合了go语言输入相关教程内容,阅读专题下面的文章了解更多详细内容。

25

2026.01.31

golang 循环遍历
golang 循环遍历

本专题整合了golang循环遍历相关教程,阅读专题下面的文章了解更多详细内容。

10

2026.01.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 5.1万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

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

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