假设一个对象有一个Finalize()
方法。
首次创建时,添加了终结队列中的指针。
该对象没有引用。
当发生垃圾收集时,它会将引用从最终化队列移动到f-reachable队列,并启动一个线程来运行Finalize
方法(顺序执行其他对象的Finalize
方法之后)
所以现在的对象(复活后)只有一个根,它是来自f-reachable队列的指针。
此时,对象是否被提升为下一代?
答案 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来回收需要完成的对象使用的内存。实际上,可能需要两个以上的集合,因为对象可以升级到老一代。