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