高效的C編程之:Switch語(yǔ)句
掃描二維碼
隨時(shí)隨地手機(jī)看文章
14.6Switch語(yǔ)句
編譯器通常將C語(yǔ)言中的Switch語(yǔ)句編譯一個(gè)查找表(TableLookup)以便跳轉(zhuǎn)到合適的入口處。
下面的例子顯示了編譯器如何處理程序中的Switch語(yǔ)言的。
C源程序如下。
char*ConditionStr1(intcondition)
{
switch(condition)
{
case0:return"EQ";
case1:return"NE";
case2:return"CS";
case3:return"CC";
case4:return"MI";
case5:return"PL";
case6:return"VS";
case7:return"VC";
case8:return"HI";
case9:return"LS";
case10:return"GE";
case11:return"LT";
case12:return"GT";
case13:return"LE";
case14:return"";
default:return0;
}
}
編譯后的結(jié)果如下。
ConditionStr1:
0000807CE1A01000MOVr1,r0
>>>SWITCH\#3switch(condition)
00008080E351000ECMPr1,#0xe
00008084908FF101ADDLSpc,pc,r1,LSL#2
00008088EA00003BB0x817c<SWITCH\#20>
0000808CEA00000DB0x80c8<SWITCH\#5>
00008090EA00000FB0x80d4<SWITCH\#6>
00008094EA000011B0x80e0<SWITCH\#7>
00008098EA000013B0x80ec<SWITCH\#8>
0000809CEA000015B0x80f8<SWITCH\#9>
000080A0EA000017B0x8104<SWITCH\#10>
000080A4EA000019B0x8110<SWITCH\#11>
000080A8EA00001BB0x811c<SWITCH\#12>
000080ACEA00001DB0x8128<SWITCH\#13>
000080B0EA00001FB0x8134<SWITCH\#14>
000080B4EA000021B0x8140<SWITCH\#15>
000080B8EA000023B0x814c<SWITCH\#16>
000080BCEA000025B0x8158<SWITCH\#17>
000080C0EA000027B0x8164<SWITCH\#18>
000080C4EA000029B0x8170<SWITCH\#19>
對(duì)于ARM代碼,查找表的入口為4字節(jié);Thumb代碼的查找表入口為1或2個(gè)字節(jié)(當(dāng)Case情況小于32時(shí),使用入口為1字節(jié)的查找表)。所以當(dāng)使用Switch語(yǔ)句時(shí),應(yīng)盡量較少Case分支。
另外,為了提高系統(tǒng)性能,也可以手工編寫(xiě)代碼,形成程序跳轉(zhuǎn)來(lái)避免使用Switch語(yǔ)句。
下面的例子顯示對(duì)上面Switch分支語(yǔ)句的改寫(xiě)。
char*ConditionStr2(intcondition)
{
if((unsigned)condition>=15)return0;
return
"EQ\0NE\0CS\0CC\0MI\0PL\0VS\0VC\0HI\0LS\0GE\0LT\0GT\0LE\0\0"+
3*condition;
}
編譯后的代碼如下所示。
ConditionStr2:
00008188E1A01000MOVr1,r0
>>>SWITCH\#26if((unsigned)condition>=15)return0;
0000818CE351000FCMPr1,#0xf
000081903A000001BCC0x819c<SWITCH\#27>
>>>SWITCH\#26if((unsigned)condition>=15)return0;
00008194E3A00000MOVr0,#0
>>>SWITCH\#30}
00008198E12FFF1EBXr14
>>>SWITCH\#26if((unsigned)condition>=15)return0;
>>>SWITCH\#27return
0000819CE28F005CADRr0,{pc}+0x64;#0x8200
000081A0E3A02003MOVr2,#3
000081A4E0200291MLAr0,r1,r2,r0
000081A8EAFFFFFAB0x8198<SWITCH\#30>
>>>SWITCH\#33{
從兩段匯編代碼的分析可以看出,使用跳轉(zhuǎn)表需要240bytes,而第二種做法只用了72bytes。