《 C 語(yǔ)言的一些“騷操作”及其深層理解》之隱藏的死循環(huán)
隱藏的死循環(huán)
有些時(shí)候我們會(huì)發(fā)現(xiàn)for循環(huán)變成了一個(gè)死循環(huán):
unsigned char i;
for(i=4;i>=0;i--) ....
我們本希望循環(huán)5次,然后結(jié)束,但是實(shí)際情況是陷入了死循環(huán)。這種錯(cuò)誤在實(shí)際開(kāi)發(fā)中,還比較難發(fā)現(xiàn)。其原因在于i的類(lèi)型,無(wú)符號(hào)整型是永遠(yuǎn)不小于0的。我們需要將i的類(lèi)型改為有符號(hào)型。
signed char i;
for(i=4;i>=0;i--) ....
OK,這樣就對(duì)了。細(xì)節(jié)雖小,但是對(duì)實(shí)際開(kāi)發(fā)的影響還是蠻大的,請(qǐng)大家引以為戒。
下面的兩個(gè)例子中for循環(huán)也是死循環(huán),請(qǐng)自行分析:
例1:
unsigned char i;
for(i=0;i<256;i++) ...
提示:i的數(shù)據(jù)類(lèi)型。
例2:
char str[20];
char *p;
unsigned char n=0;
for(p=strcpy(str," abcd");((*p)=' ');p++,n++);
提示:這個(gè)例子,不光會(huì)死循環(huán),而且還可能會(huì)讓程序直接崩潰。判等的==你會(huì)不會(huì)經(jīng)常直接寫(xiě)錯(cuò)成=(賦值表達(dá)式)。
看似多余的空循環(huán)
有時(shí)我們會(huì)看到這樣的代碼:
do
{
...... //do something
}while(0);
代碼本身實(shí)際只運(yùn)行了一次,為什么要在它外面加一層do while呢?這看似是多余的。其實(shí)不然,我們來(lái)看下面例子:
#define DO_SOMETHING fun1();fun2();
void main(void)
{
while(1) DO_SOMETHING;
}
while(1) DO_SOMETHING;本意應(yīng)該是不斷調(diào)用fun1和fun2,但實(shí)際上只有fun1得到運(yùn)行。其中原因大家應(yīng)該明白。所以,我們可以這樣來(lái)寫(xiě):
#define DO_SOMETHING do{ fun1();fun2();}while(0);
do while就如同一個(gè)框架把要運(yùn)行的代碼框起來(lái),成為一個(gè)整體。
獨(dú)立執(zhí)行體
我在C語(yǔ)言編程的過(guò)程中,經(jīng)常樂(lè)于使用一種“局部獨(dú)立化”的方式,我稱(chēng)之為“獨(dú)立執(zhí)行體”,如下例:
void fun(int a,int b,int c)
{
Int tmp=0;
//主體計(jì)算
{ //獨(dú)立執(zhí)行體,解決臨時(shí)性問(wèn)題
int c=0;
c=(a>b)?a:b;
printf("max:%d\r\n",c);
}
{ //獨(dú)立執(zhí)行體
int c=0,d=0,.....,res=0.;
//數(shù)據(jù)處理算法
printf("result:%d\r\n",res);
}
//進(jìn)一步計(jì)算
}
編程時(shí),我們經(jīng)常需要解決一些小問(wèn)題,比如想對(duì)一些數(shù)據(jù)進(jìn)行臨時(shí)性的處理,查看中間結(jié)果;或是臨時(shí)性的突發(fā)奇想,試探性的作一些小算法。這過(guò)程中可能需要獨(dú)立的變量,以及獨(dú)立于主體程序的執(zhí)行邏輯,但又覺(jué)得不至于去專(zhuān)門(mén)定義一個(gè)函數(shù),只是想一帶而過(guò)。比如上例,函數(shù)fun主要對(duì)a、b、c這3個(gè)參數(shù)進(jìn)行計(jì)算(使用某種算法),過(guò)程中想臨時(shí)看一下a和b誰(shuí)比較大,由第一個(gè)“獨(dú)立執(zhí)行體”來(lái)完成,其中的代碼由自己的{}擴(kuò)起來(lái)。
其實(shí)我們可以更深層的去理解C語(yǔ)言中的{},它為我們開(kāi)辟了一個(gè)可自由編程的獨(dú)立空間。在{}里,可以定義變量,可以調(diào)用函數(shù)以及訪問(wèn)外層代碼中的變量,可以作宏定義等等。平時(shí)我們使用的函數(shù),它的{}部分其實(shí)就是一個(gè)“獨(dú)立執(zhí)行體”。
“獨(dú)立執(zhí)行體”的思想,也許可以讓我們編程更加靈活方便,可以隨時(shí)讓我們直接得到一塊自由編程的靜土。
上一節(jié)中的do while(0),其實(shí)完全可以把do while(0)去掉,只用{}即可:
#define DO_SOMETHING {fun1();fun2();}
其中它還有一個(gè)好處,就是當(dāng)你不需要這段代碼的時(shí)候,你可以直接在{}前面加上if(0)即可。一個(gè)“獨(dú)立執(zhí)行體”的外層是可以受if、do while、while、for等這些條件控制的。