给出以下代码:
namespace GcDemo
{
class Program
{
static void Main(string[] args)
{
var list = new List<object>();
Console.WriteLine("list is in {0} generation.", GC.GetGeneration(list));
GC.Collect();
Console.WriteLine("list is in {0} generation.", GC.GetGeneration(list));
GC.Collect();
list.Add(new object());
Console.WriteLine("list is in {0} generation. object is in {1} generation.", GC.GetGeneration(list), GC.GetGeneration(list[0]));
GC.Collect(0);
Console.WriteLine("list is in {0} generation. object is in {1} generation.", GC.GetGeneration(list), GC.GetGeneration(list[0]));
}
}
}
列表对象是第 2 代中的,而它是第 0 代中唯一引用的 list[0] 对象。它为什么知道不在 GC.Collect(0) 中收集它?
最佳答案
This article有一个简单的说明它是如何工作的;阅读“让世代与写入障碍一起工作”部分。 This是另一篇很好的博客文章,更详细地解释了该技术。
执行摘要是 CLR 发出代码,以便它可以检测 Gen2 对象何时被写入。它将写入记录到数据结构(“卡表”)中,并在执行 Gen0 收集时检查该数据结构;这使得它能够找到 Gen2 -> Gen0 引用,而无需花费遍历内存中所有对象的全部成本。
关于c# - GC 世代 - 收集问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12809365/