ยังไม่มีข้อมูลเกี่ยวกับสิ่งที่เกิดขึ้นจริงใต้เข็มขัด ดูตัวอย่างนี้:
object o = "test";
if (o is string)
{
var x = (string) o;
}
สิ่งนี้แปลเป็น IL ต่อไปนี้:
IL_0000: nop
IL_0001: ldstr "test"
IL_0006: stloc.0 // o
IL_0007: ldloc.0 // o
IL_0008: isinst System.String
IL_000D: ldnull
IL_000E: cgt.un
IL_0010: stloc.1
IL_0011: ldloc.1
IL_0012: brfalse.s IL_001D
IL_0014: nop
IL_0015: ldloc.0 // o
IL_0016: castclass System.String
IL_001B: stloc.2 // x
IL_001C: nop
IL_001D: ret
สิ่งสำคัญที่นี่คือisinst
และการcastclass
โทร - ทั้งที่ค่อนข้างแพง หากคุณเปรียบเทียบสิ่งนั้นกับทางเลือกอื่นคุณจะเห็นว่ามีเพียงการisinst
ตรวจสอบ:
object o = "test";
var oAsString = o as string;
if (oAsString != null)
{
}
IL_0000: nop
IL_0001: ldstr "test"
IL_0006: stloc.0 // o
IL_0007: ldloc.0 // o
IL_0008: isinst System.String
IL_000D: stloc.1 // oAsString
IL_000E: ldloc.1 // oAsString
IL_000F: ldnull
IL_0010: cgt.un
IL_0012: stloc.2
IL_0013: ldloc.2
IL_0014: brfalse.s IL_0018
IL_0016: nop
IL_0017: nop
IL_0018: ret
สิ่งที่ควรค่าแก่การกล่าวถึงคือประเภทค่าจะใช้unbox.any
มากกว่าcastclass
:
object o = 5;
if (o is int)
{
var x = (int)o;
}
IL_0000: nop
IL_0001: ldc.i4.5
IL_0002: box System.Int32
IL_0007: stloc.0 // o
IL_0008: ldloc.0 // o
IL_0009: isinst System.Int32
IL_000E: ldnull
IL_000F: cgt.un
IL_0011: stloc.1
IL_0012: ldloc.1
IL_0013: brfalse.s IL_001E
IL_0015: nop
IL_0016: ldloc.0 // o
IL_0017: unbox.any System.Int32
IL_001C: stloc.2 // x
IL_001D: nop
IL_001E: ret
แต่ทราบว่าเรื่องนี้ไม่จำเป็นต้องแปลว่าจะได้เร็วขึ้นส่งผลให้ในขณะที่เราสามารถดูได้ที่นี่ ดูเหมือนจะมีการปรับปรุงตั้งแต่คำถามที่ถูกถามว่า: บรรยากาศดูเหมือนจะดำเนินการให้เร็วที่สุดเท่าที่พวกเขาเคยเป็น แต่as
และlinq
ในขณะนี้มีประมาณ 3 ครั้งได้เร็วขึ้น
MyProp
หลังจากการเปลี่ยนแปลงนี้