复活后的GC生成?

时间:2012-10-27 20:12:53

标签: c# .net garbage-collection

假设一个对象有一个Finalize()方法。

首次创建时,添加了终结队列中的指针。

该对象没有引用。

当发生垃圾收集时,它会将引用从最终化队列移动到f-reachable队列,并启动一个线程来运行Finalize方法(顺序执行其他对象的Finalize方法之后)

所以现在的对象(复活后)只有一个根,它是来自f-reachable队列的指针。

此时,对象是否被提升为下一代?

2 个答案:

答案 0 :(得分:3)

这是你可以尝试的东西。在没有附加调试器的Release版本中运行此代码:

using System;

class Program {
    static void Main(string[] args) {
        var obj = new Foo();
        // There are 3 generations, could have used GC.MaxGeneration
        for (int ix = 0; ix < 3; ++ix) {
            GC.Collect();
            GC.WaitForPendingFinalizers();
        }
        Console.ReadLine();
    }
}

class Foo {
    int attempt = 0;
    ~Foo() {
        // Don't try to do this over and over again, rough at program exit
        if (attempt++ < 3) {
            GC.ReRegisterForFinalize(this);
            Console.WriteLine(GC.GetGeneration(this));
        }
    }
}

输出:

  

1
  2
  2个

所以它停留在它被集合移动到的那一代,移动到每个集合的下一个集合,直到它到达最后一个集合。这一般是有道理的。

答案 1 :(得分:1)

似乎答案是肯定的,这将会发生。 http://msdn.microsoft.com/en-us/magazine/bb985010.aspx说:

  

...需要两个GC来回收需要完成的对象使用的内存。实际上,可能需要两个以上的集合,因为对象可以升级到老一代