कंपाइलर बग।
जानकारी के लिए, मेरे पास async सीटीपी है, जो संबंधित हो सकता है; csc
रिपोर्ट: 4.0.30319.440
के बीच एक अंतर लगता है:
.method public hidebysig static void TryTwo() cil managed
{
.maxstack 4
.locals init (
[0] class Line <>g__initLocal6)
L_0000: newobj instance void Line::.ctor()
L_0005: stloc.0
L_0006: ldloc.0
L_0007: ldc.i4.3
L_0008: newobj instance void [mscorlib]System.Decimal::.ctor(int32)
L_000d: stfld valuetype [mscorlib]System.Decimal Line::A
L_0012: ldloc.0
L_0013: ldc.i4.5
L_0014: newobj instance void [mscorlib]System.Decimal::.ctor(int32)
L_0019: stfld valuetype [mscorlib]System.Decimal Line::B
L_001e: ldsfld int32[] Test::integers
L_0023: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_0028: brtrue.s L_003b
L_002a: ldnull
L_002b: ldftn int32 Test::<TryTwo>b__7(int32)
L_0031: newobj instance void [mscorlib]System.Func`2<int32, int32>::.ctor(object, native int)
L_0036: stsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_003b: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_0040: call class [mscorlib]System.Collections.Generic.Dictionary`2<!!1, !!2> [System.Core]System.Linq.Enumerable::ToDictionary<int32, int32, valuetype [mscorlib]System.Decimal>(class [mscorlib]System.Collections.Generic.IEnumerable`1<!!0>, class [mscorlib]System.Func`2<!!0, !!1>, class [mscorlib]System.Func`2<!!0, !!2>)
L_0045: pop
L_0046: ldstr "TryTwo complete"
L_004b: call void [mscorlib]System.Console::WriteLine(string)
L_0050: ret
}
बनाम
.method public hidebysig static void TryFive() cil managed
{
.maxstack 4
.locals init (
[0] class Line line,
[1] class [mscorlib]System.Func`2<int32, valuetype [mscorlib]System.Decimal> func,
[2] class Line <>g__initLocal9)
L_0000: newobj instance void Line::.ctor()
L_0005: stloc.2
L_0006: ldloc.2
L_0007: ldc.i4.3
L_0008: newobj instance void [mscorlib]System.Decimal::.ctor(int32)
L_000d: stfld valuetype [mscorlib]System.Decimal Line::A
L_0012: ldloc.2
L_0013: ldc.i4.5
L_0014: newobj instance void [mscorlib]System.Decimal::.ctor(int32)
L_0019: stfld valuetype [mscorlib]System.Decimal Line::B
L_001e: ldloc.2
L_001f: stloc.0
L_0020: ldloc.0
L_0021: ldftn valuetype [mscorlib]System.Decimal SimpleCompute::Compute(class Line, int32)
L_0027: newobj instance void [mscorlib]System.Func`2<int32, valuetype [mscorlib]System.Decimal>::.ctor(object, native int)
L_002c: stloc.1
L_002d: ldsfld int32[] Test::integers
L_0032: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegateb
L_0037: brtrue.s L_004a
L_0039: ldnull
L_003a: ldftn int32 Test::<TryFive>b__a(int32)
L_0040: newobj instance void [mscorlib]System.Func`2<int32, int32>::.ctor(object, native int)
L_0045: stsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegateb
L_004a: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegateb
L_004f: ldloc.1
L_0050: call class [mscorlib]System.Collections.Generic.Dictionary`2<!!1, !!2> [System.Core]System.Linq.Enumerable::ToDictionary<int32, int32, valuetype [mscorlib]System.Decimal>(class [mscorlib]System.Collections.Generic.IEnumerable`1<!!0>, class [mscorlib]System.Func`2<!!0, !!1>, class [mscorlib]System.Func`2<!!0, !!2>)
L_0055: pop
L_0056: ldstr "TryFour complete"
L_005b: call void [mscorlib]System.Console::WriteLine(string)
L_0060: ret
}
आप में देखो, तो:
public static void TryTwo() // fails
{
var line = new Line {A = 3, B = 5};
var dict = integers.ToDictionary<int, int, decimal>(i => i, line.Compute);
Console.WriteLine("TryTwo complete");
}
public static void TryFive() // works
{
var line = new Line { A = 3, B = 5 };
Func<int, decimal> func = line.Compute;
var dict = integers.ToDictionary<int, int, decimal>(i => i, func);
Console.WriteLine("TryFour complete");
}
तो परावर्तक में नजर डालते हैं टूटा संस्करण, यह केवल एक प्रतिनिधि लोड करता है। संकलक बग, मूल रूप से:
L_0023: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_0028: brtrue.s L_003b
L_002a: ldnull
L_002b: ldftn int32 Test::<TryTwo>b__7(int32)
L_0031: newobj instance void [mscorlib]System.Func`2<int32, int32>::.ctor(object, native int)
L_0036: stsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_003b: ldsfld class [mscorlib]System.Func`2<int32, int32> Test::CS$<>9__CachedAnonymousMethodDelegate8
L_0040: call class [mscorlib]System.Collections.Generic.Dictionary`2<!!1, !!2> [System.Core]System.Linq.Enumerable::ToDictionary<int32, int32, valuetype [mscorlib]System.Decimal>(class [mscorlib]System.Collections.Generic.IEnumerable`1<!!0>, class [mscorlib]System.Func`2<!!0, !!1>, class [mscorlib]System.Func`2<!!0, !!2>)
ऊपर की
सब है "जाँच करें कि क्या कैश्ड i => i
मौजूद है, अगर यह नहीं बना, तो इसे लोड"। यह दूसरे प्रतिनिधि के साथ कुछ भी नहीं करता है। नतीजतन, विधि कॉल करने के लिए ढेर पर पर्याप्त मूल्य नहीं हैं।
असल में, मेरे स्थानीय इंस्टॉल पर, आपकी "पास" लाइन भी –
विफल हो जाती है यह मेरे (v4.0, x86) बॉक्स पर ठीक काम कर रहा है - .NET का कौन सा संस्करण आप चल रहे हैं, और क्या आर्किटेक्चर? –
दिलचस्प! यदि आप एक्सटेंशन विधि को हटाते हैं (यानी सरल कॉम्प्यूट को लाइन पर एक इंस्टेंस विधि के रूप में रखें) क्या यह काम करता है? अगर मैं ऐसा करता हूं तो यह मेरे लिए करता है? –