我可以比较IL代码来确定哪种技术更快或更好吗?

时间:2009-08-19 00:48:34

标签: c# performance linqpad

背景

This question让我想到了什么。最近,由于我一直在研究 linq pad 的IL功能,我一直在将两种方法的IL代码与同一问题进行比较,以“确定”最佳。< / p>

使用上面链接的问题,关于转换数组,我为两个答案生成了IL代码:

var arr = new string[] { "1", "2", "3", "4" };
var result = Array.ConvertAll(arr, s => Int32.Parse(s));

制备:

IL_0001:  ldc.i4.4    
IL_0002:  newarr      System.String
IL_0007:  stloc.2     
IL_0008:  ldloc.2     
IL_0009:  ldc.i4.0    
IL_000A:  ldstr       "1"
IL_000F:  stelem.ref  
IL_0010:  ldloc.2     
IL_0011:  ldc.i4.1    
IL_0012:  ldstr       "2"
IL_0017:  stelem.ref  
IL_0018:  ldloc.2     
IL_0019:  ldc.i4.2    
IL_001A:  ldstr       "3"
IL_001F:  stelem.ref  
IL_0020:  ldloc.2     
IL_0021:  ldc.i4.3    
IL_0022:  ldstr       "4"
IL_0027:  stelem.ref  
IL_0028:  ldloc.2     
IL_0029:  stloc.0     
IL_002A:  ldloc.0     
IL_002B:  ldsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_0030:  brtrue.s    IL_0045
IL_0032:  ldnull      
IL_0033:  ldftn       b__0
IL_0039:  newobj      System.Converter<System.String,System.Int32>..ctor
IL_003E:  stsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_0043:  br.s        IL_0045
IL_0045:  ldsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_004A:  call        System.Array.ConvertAll
IL_004F:  stloc.1     

b__0:
IL_0000:  ldarg.0     
IL_0001:  call        System.Int32.Parse
IL_0006:  stloc.0     
IL_0007:  br.s        IL_0009
IL_0009:  ldloc.0     
IL_000A:  ret         

和另一个答案:

var arr = new string[] { "1", "2", "3", "4" };
var result = arr.Select(s => int.Parse(s)).ToArray();

制备:

IL_0001:  ldc.i4.4    
IL_0002:  newarr      System.String
IL_0007:  stloc.2     
IL_0008:  ldloc.2     
IL_0009:  ldc.i4.0    
IL_000A:  ldstr       "1"
IL_000F:  stelem.ref  
IL_0010:  ldloc.2     
IL_0011:  ldc.i4.1    
IL_0012:  ldstr       "2"
IL_0017:  stelem.ref  
IL_0018:  ldloc.2     
IL_0019:  ldc.i4.2    
IL_001A:  ldstr       "3"
IL_001F:  stelem.ref  
IL_0020:  ldloc.2     
IL_0021:  ldc.i4.3    
IL_0022:  ldstr       "4"
IL_0027:  stelem.ref  
IL_0028:  ldloc.2     
IL_0029:  stloc.0     
IL_002A:  ldloc.0     
IL_002B:  ldsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_0030:  brtrue.s    IL_0045
IL_0032:  ldnull      
IL_0033:  ldftn       b__0
IL_0039:  newobj      System.Func<System.String,System.Int32>..ctor
IL_003E:  stsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_0043:  br.s        IL_0045
IL_0045:  ldsfld      UserQuery.CS$<>9__CachedAnonymousMethodDelegate1
IL_004A:  call        System.Linq.Enumerable.Select
IL_004F:  call        System.Linq.Enumerable.ToArray
IL_0054:  stloc.1     

b__0:
IL_0000:  ldarg.0     
IL_0001:  call        System.Int32.Parse
IL_0006:  stloc.0     
IL_0007:  br.s        IL_0009
IL_0009:  ldloc.0     
IL_000A:  ret     

看看这个,我能说的是后一个选项

  • 需要额外增加一行
  • 在第一个答案没有
  • 时使用linq
  • 通过IL_0039以不同方式创建Int。

问题

  • 对于这个具体的例子,我的假设是否正确?
  • 一般来说,我应该如何通过IL代码比较两个解决方案?
  • 一般来说,IL LOC较少的解决方案是否意味着它会更快或使用更少的内存?
  • 正如标题所说,我可以比较IL代码以确定哪种技术更快或更好吗?

FWIW,我不会经常这样做,只是在极少数情况下,当开发人员在工作中进行一些讨论时。有人会说“哦这更有效”,我们会把它扔进linqpad来查看IL代码。 FWIW,在获得高效/快速的方法之前,我几乎总是遵守它的工作原理。正因如此,人们不认为我一直在比较我正在开发的IL代码:)

2 个答案:

答案 0 :(得分:8)

  
      
  • 对于这个具体的例子,我的假设是否正确?
  •   
  • 一般来说,我应该如何通过IL代码比较两个解决方案?
  •   
  • 一般来说,IL LOC较少的解决方案是否意味着它会更快或使用更少的内存?
  •   
  • 正如标题所说,我可以比较IL代码以确定哪种技术更快或更好吗?
  •   

1)你的假设对于正在发生的事情是正确的。

2)您需要了解IL代码正在做什么才能确定哪个是“更好”

3)否。这意味着它需要更少的指令才能运行。但是,这些单独的指令可能会使用更多内存或更少。例如,您引用的指令(在一种情况下是创建Func委托),另一种情况是创建Converter对象。没有更多信息,很难说这两件事中哪一件更贵。

4)是的,不是......

问题是IL代码会告诉你发生了什么,但实际上IL中的嵌套调用将成为大型性能驱动程序。如果IL代码在任何地方都进行简单操作,通常越短越好(尽管单个IL操作的速度本身可能不同)。当代码调用其他类型的方法或构造函数时,例如你的代码或构造函数,就不可能单独告诉它。在一种情况下,IL的一行可能需要更长的时间(例如,如果它调用的是昂贵的方法),而在另一种情况下(在这种情况下,他们正在进行简单的操作)。

在上面的例子中,例如,前20个操作非常非常快,而最后几个操作几乎占用了所有可执行时间。

答案 1 :(得分:3)

两个答案的工作大块在IL 004A(和第二个的IL 004F)完成。除非你知道这些外部电话的费用,否则你没有实际可以比较两个答案的表现。