19

有什么区别

Class1.Method1<Guid, BECustomer>("cId", Facade.Customers.GetSingle);

Class1.Method1<Guid, BECustomer>("cId", x => Facade.Customers.GetSingle(x));

?

Resharper 建议使用第一个表达式。

4

3 回答 3

23

结果没有区别。但是,第二个创建了一个额外的重定向:代码将首先调用您的匿名方法,该方法接受一个名为的参数x,然后Facade.Customers.GetSingle使用该参数调用。这种重定向根本没有任何好处,这就是 ReSharper 告诉您使用第一个替代方案的原因。

于 2011-07-12T10:11:41.997 回答
16

在幕后,如果您使用 lambda 表达式,编译器会生成更多代码。使用方法组,它只是创建一个指向该方法的新委托:

L_0001: ldstr "cId"
L_0006: ldnull 
L_0007: ldftn void Facade/Customers::GetSingle(valuetype [mscorlib]System.Guid)
L_000d: newobj instance void [mscorlib]System.Action`1<valuetype [mscorlib]System.Guid>::.ctor(object, native int)
L_0012: call void Class1::Method1<valuetype [mscorlib]System.Guid, class BECustomer>(string, class [mscorlib]System.Action`1<!!0>)

使用 lambda 表达式,会在类(在 L_0025 上)上创建一个匿名方法<Test>b__0,而委托则引用它:

L_0018: ldstr "cId"
L_001d: ldsfld class [mscorlib]System.Action`1<valuetype [mscorlib]System.Guid> Class1::CS$<>9__CachedAnonymousMethodDelegate1
L_0022: brtrue.s L_0037
L_0024: ldnull 
L_0025: ldftn void Class1::<Test>b__0(valuetype [mscorlib]System.Guid)
L_002b: newobj instance void [mscorlib]System.Action`1<valuetype [mscorlib]System.Guid>::.ctor(object, native int)
L_0030: stsfld class [mscorlib]System.Action`1<valuetype [mscorlib]System.Guid> Class1::CS$<>9__CachedAnonymousMethodDelegate1
L_0035: br.s L_0037
L_0037: ldsfld class [mscorlib]System.Action`1<valuetype [mscorlib]System.Guid> Class1::CS$<>9__CachedAnonymousMethodDelegate1
L_003c: call void Class1::Method1<valuetype [mscorlib]System.Guid, class BECustomer>(string, class [mscorlib]System.Action`1<!!0>)
于 2011-07-12T10:24:26.870 回答
1

Method1<Guid, BECustomer>接受一个Func<Guid, BECustomer>论点的地方Func<Guid, BECustomer>是:

public delegate BECustomer Func(Guid arg);

事实上,所有的 aFunc都是一个泛型委托:

public delegate TResult Func<T, TResult>(T arg);

编译器可以分析您的代码并确定您Func<Guid, BECustomer>与方法组兼容,Facade.Customers.GetSingle因为方法签名与委托签名匹配。

这是语法糖,是编译器为您完成繁重工作的另一个示例。

于 2011-07-12T10:17:45.333 回答