用于基准测试目的的C#编译器优化

时间:2011-06-29 16:32:48

标签: c# .net optimization profiling benchmarking

我需要检查一些.NET API的性能,然后我想出了这段代码。

Stopwatch sw = Stopwatch.StartNew();
for (int i = 0; i < 1000; i++) {
    int res = (int) SOME .NET FUNCTION TO RUN;
}
sw.Stop();
double time = sw.Elapsed.TotalMilliseconds;

在我的第二个想法中,我担心编译器会优化所有操作,因为res没有在任何地方使用。我修改了代码如下。

Stopwatch sw = Stopwatch.StartNew();
long sumIt = 0;
for (int i = 0; i < 1000; i++) {
    int res = (int) SOME .NET FUNCTION TO RUN;
    sumIt += res;
}
sw.Stop();
double time = sw.Elapsed.TotalMilliseconds;
value = (int) sumIt / 1000;

有趣的是,编译器似乎没有用我的第一个例子来优化操作。我用csc(Visual Studio)和mono。

进行了测试

这是我的问题。

  • 作为我的第一个例子制作基准是否安全?
  • C#编译器是否足够聪明,无法理解某些代码可以优化出来?或者,是否有任何编译器参数?

ADDED

SOME FUNCTION TO RUN实际上是SOME .NET FUNCTION TO RUN,我修改了OP。 根据答案,看起来C#编译器不能(或不)优化操作,因为.NET FUNCTION TO RUN可能有副作用。

1 个答案:

答案 0 :(得分:2)

编译器(JIT)可以在发现它没有副作用时优化整个函数调用。它可能需要能够内联函数来检测它。

我尝试了只对输入参数起作用的小函数,看看它是通过检查结果汇编来优化的(确保在未选中的情况下尝试使用“模块加载上的Suppres优化”的版本构建)。

 ...
 for (int i = 0; i < 1000; i++) 
 {    
    int res = (int) Func(i);
 }
 ...

 static int Func(int arg1)
 {
    return arg1 * arg1;
 }

拆卸:

      for (int i = 0; i < 1000; i++) 
    00000016  xor         eax,eax 
    00000018  inc         eax 
    00000019  cmp         eax,3E8h 
    0000001e  jl          00000018 
        }