GC.Collect() and Finalize(GC.Collect() 和完成)
问题描述
好的,众所周知,当 GC 将对象识别为垃圾时,它会隐式调用对象的 Finalize
方法.但是如果我执行 GC.Collect()
会发生什么?终结器是否仍在执行?可能是个愚蠢的问题,但有人问我这个问题,我回答是",然后我想:这完全正确吗?"
Ok, it's known that GC implicitly calls Finalize
methods on objects when it identifies that object as garbage. But what happens if I do a GC.Collect()
? Are the finalizers still executed? A stupid question maybe, but someone asked me this and I answered a "Yes" and then I thought: "Was that fully correct?"
推荐答案
好的,众所周知,当 GC 将对象识别为垃圾时,它会隐式调用对象的 Finalize 方法.
Ok, it's known that GC implicitly calls Finalize methods on objects when it identifies that object as garbage.
不不不.这不是已知,因为要成为knowledge,一个陈述必须是true.该陈述是错误.垃圾收集器在跟踪时不会运行终结器,无论它是自行运行还是您调用 Collect
.终结器线程在跟踪收集器找到垃圾后运行终结器,并且对于 Collect
的调用异步发生.(如果它发生了,它可能不会发生,正如另一个答案指出的那样.)也就是说,你不能依赖在控制从 Collect
返回之前执行的终结器线程.
No no no. That is not known because in order to be knowledge a statement must be true. That statement is false. The garbage collector does not run finalizers as it traces, whether it runs itself or whether you call Collect
. The finalizer thread runs finalizers after the tracing collector has found the garbage and that happens asynchronously with respect to a call to Collect
. (If it happens at all, which it might not, as another answer points out.) That is, you cannot rely on the finalizer thread executing before control returns from Collect
.
以下是其工作原理的简化草图:
Here's an oversimplified sketch of how it works:
- 当收集发生时,垃圾收集器跟踪线程会跟踪根——已知的活动对象,以及它们引用的每个对象,等等——以确定死对象.
- 具有挂起终结器的死"对象被移动到终结器队列中.终结器队列是根.因此,那些死"的物体实际上还活着.
- 终结器线程(通常是与 GC 跟踪线程不同的线程)最终运行并清空终结器队列.然后这些对象就真正死了,并被收集到跟踪线程上的 next 集合中.(当然,由于它们刚刚在第一个系列中幸存下来,它们可能处于更高的一代.)
- When a collection happens the garbage collector tracing thread traces the roots -- the objects known to be alive, and every object they refer to, and so on -- to determine the dead objects.
- "Dead" objects that have pending finalizers are moved onto the finalizer queue. The finalizer queue is a root. Therefore those "dead" objects are actually still alive.
- The finalizer thread, which is typically a different thread than the GC tracing thread, eventually runs and empties out the finalizer queue. Those objects then become truly dead, and are collected in the next collection on the tracing thread. (Of course, since they just survived the first collection, they might be in a higher generation.)
正如我所说,这过于简单了;终结器队列如何工作的确切细节比这要复杂一些.但它已经足够传达这个想法了.这里的实际结果是 您不能假设调用 Collect
也会运行终结器,因为它不会.让我再重复一遍:垃圾收集器的跟踪部分不运行终结器,Collect
只运行收集机制.
As I said, that's oversimplified; the exact details of how the finalizer queue works are a bit more complicated than that. But it gets enough of the idea across. The practical upshot here is that you cannot assume that calling Collect
also runs finalizers, because it doesn't. Let me repeat that one more time: the tracing portion of the garbage collector does not run finalizers, and Collect
only runs the tracing part of the collection mechanism.
如果您想保证所有终结器都已运行,请在调用 Collect
之后调用恰当命名的 WaitForPendingFinalizers
.这将暂停当前线程,直到终结器线程开始清空队列.如果你想确保这些最终对象的内存被回收,那么你将不得不调用 Collect
秒次.
Call the aptly named WaitForPendingFinalizers
after calling Collect
if you want to guarantee that all finalizers have run. That will pause the current thread until the finalizer thread gets around to emptying the queue. And if you want to ensure that those finalized objects have their memory reclaimed then you're going to have to call Collect
a second time.
当然,不言而喻,您应该只出于调试和测试目的而这样做.千万不要在没有真正真正充分理由的情况下在生产代码中做这种废话.
And of course, it goes without saying that you should only be doing this for debugging and testing purposes. Never do this nonsense in production code without a really, really good reason.
这篇关于GC.Collect() 和完成的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:GC.Collect() 和完成
- Web Api 中的 Swagger .netcore 3.1,使用 swagger UI 设置日期时间格式 2022-01-01
- WebMatrix WebSecurity PasswordSalt 2022-01-01
- 带有服务/守护程序应用程序的 Microsoft Graph CSharp SDK 和 OneDrive for Business - 配额方面返回 null 2022-01-01
- 输入按键事件处理程序 2022-01-01
- C# 中多线程网络服务器的模式 2022-01-01
- C#MongoDB使用Builders查找派生对象 2022-09-04
- MoreLinq maxBy vs LINQ max + where 2022-01-01
- 如何用自己压缩一个 IEnumerable 2022-01-01
- 良好实践:如何重用 .csproj 和 .sln 文件来为 CI 创建 2022-01-01
- 在哪里可以找到使用中的C#/XML文档注释的好例子? 2022-01-01