测量“在 GC 中花费的时间百分比”

Measuring "% Time spent in GC"

我正在尝试测量 GC 我的应用程序中两点之间花费了多少时间。像这样:

GC.Collect();
GC.WaitForPendingFinalizers();
GC.Collect();
gcPerfCounter.NextValue();

// This is the part that I would like to measure
doStuff();

var timeSpentInGC = gcPerfCounter.NextValue();

但是,我不确定这是使用该计数器的正确方法。在这个 post: Correct use of "% Time in GC" performance counter 中,答案说它应该像秒表一样使用,这就是我在操作前调用 NextValue() 的原因。我不认为是这种情况,因为当我删除第一个 gcPerfCounter.NextValue() 行时它给了我相同的结果。

MSDN 文章提到“GC 时间百分比”性能计数器:

Displays the percentage of elapsed time that was spent performing a garbage collection since the last garbage collection cycle. https://msdn.microsoft.com/en-us/library/w8f5kw2e(v=vs.110).aspx

这让我感到困惑,因为我认为自从上次 GC cycle 以来我们在 GC 中花费的时间为零。所以根据这个描述,这个值应该永远为零。

你能解释一下这个性能计数器在我的例子中的含义和正确用法吗?

我同意性能计数器的描述令人困惑。为了更好地理解这里的意义,请参阅 Performance Counters in the .NET Framework 的完整描述,您提供了 link 至:

Displays the percentage of elapsed time that was spent performing a garbage collection since the last garbage collection cycle. This counter usually indicates the work done by the garbage collector to collect and compact memory on behalf of the application. This counter is updated only at the end of every garbage collection. This counter is not an average; its value reflects the last observed value.

明确指出计数器在垃圾回收结束时更新。基于这个事实,我相信您可以使用以下时间线来更好地理解计数器:

       GC0                        GC1
----+-------+------------------+-------+----
 Tstart0  Tend0             Tstart1  Tend1

            |    Not in GC     | In GC |
            +------------------+-------+

描述中提到的垃圾收集周期是区间[Tend0, TEnd1]。有了这个定义,描述中的第一句话就变得更有意义了。

计数器在每个垃圾收集周期 Tend 处计算。参考时间线下方显示的时间间隔,该值是 在 GC 中的时间占整个垃圾收集周期总时间的百分比(不在 GC在 GC).

或者更正式地说,在 Tend1 处,性能计数器的值为:

% Time in GC = 100*(Tend1 - Tstart1)/(Tend1 - Tend0)

所以计数器确实给出了 GC 所花费时间的百分比,并且对计数器进行采样将给出基于前两次 GC 执行的值。

要检查这一点,您可以尝试对计数器 % Time in GC 以及三个 # Gen X Collections 计数器和看到第一个计数器仅在其他三个计数器中的一个发生变化时发生变化(例如,在两个样本之间执行了 GC)。

此计数器可用于检查某个软件是否花费了不成比例的大量时间进行垃圾收集。但是,由于计数器仅反映上一个垃圾回收周期所花费的时间,您将不得不随着时间的推移对计数器进行采样以正确进行此检查。