糯米文學吧

位置:首頁 > 計算機 > C語言

C語言中do...while(0)技巧

C語言9.34K

在linux核心程式碼中,經常看到e(0)的巨集,e(0)有很多作用,具體技巧有哪些呢?以下僅供參考!

C語言中do...while(0)技巧

1、避免goto語句:

通常,如果一個函式開始要分配一些資源,然後如果在中途遇到錯誤則要退出函式,當然,退出前要釋放資源,我們的程式碼可能如下:

#defien N 10

bool Execute()

{

// 分配資源

int *p = (int *)malloc(N * sizeof(int));

bool bOk = true;

// 執行並進行錯誤處理

bOk = func1();

if(!bOk)

{

free(p);

p = NULL;

return false;

}

bOk = func2();

if(!bOk)

{

free(p);

p = NULL;

return false;

}

bOk = func3();

if(!bOk)

{

free(p);

p = NULL;

return false;

}

// ..........

// 執行成功,釋放資源並返回

free(p);

p = NULL;

return true;

}

這裡最大的問題是程式碼冗餘,每增加一個操作,就要做相應的錯誤處理,非常不靈活,於是想到了一下的goto:

#defien N 10

bool Execute()

{

// 分配資源

int *p = (int *)malloc(N * sizeof(int));

bool bOk = true;

// 執行並進行錯誤處理

bOk = func1();

if(!bOk) goto errorhandle;

bOk = func2();

if(!bOk) goto errorhandle;

bOk = func3();

if(!bOk) goto errorhandle;

// ..........

// 執行成功,釋放資源並返回

free(p);

p = NULL;

return true;

errorhandle:

free(p);

p = NULL;

return false;

}

程式碼冗餘是解決了,但是引入了C語言中比較微妙的goto語句,雖然正確的使用goto語句可以大大提高程式的靈活性與簡潔性,但是會使我們的程式捉摸不定,為了既避免使用goto語句,又能消除程式碼冗餘,可以考慮使用下面的` e(0):

#defien N 10

bool Execute()

{

//分配資源

int *p = (int *)malloc(N * sizeof(int));

bool bOK = true;

do {

//執行並進行錯誤處理

bOK = fun1();

if(!bOK) break;

bOK = fun2();

if(!bOK) break;

bOK = fun3();

if(!bOK) break;

//.........

} while(0);

//釋放資源

free(p);

p = NULL;

return bOK;

}

2、避免空宣告在編譯時出現警告:

在linux核心原始碼中,經常看到如下巨集以避免在編譯時出現警告:

#define FOO do { } while(0)

3、編寫符合習慣的程式碼塊:

你可能經常會使用如下的巨集:

#define exch(x,y) { int tmp; tmp=x; x=y; y=tmp; }

然而在某些情況下將會失效,下面的程式碼使用...

if (x > y)

exch(x,y); // 分支 1

else

do_something(); // 分支 2

但是將被解釋為一個分支的if語句:

if (x > y) {

int tmp;

tmp = x;

x = y;

y = tmp;

}

; // 空語句

else // ERROR!!!

do_something();

錯誤出在“;”直接位於程式碼塊的後面,解決的辦法是將程式碼嵌入e(0),於是得到下面的程式碼:

if (x > y)

do {

int tmp;

tmp = x;

x = y;

y = tmp;

} while(0);

else

do_something();

於是上面的巨集可以修改為:

#define exch(x,y) do {

int tmp;

tmp = x;

x = y;

y = tmp;

} while(0)

4、在條件語句中使用複雜的巨集:

假如一個巨集包含類似如下幾行程式碼:

#define FOO(x)

printf("arg is %s", x);

do_something_useful(x);

現在想像一下下面的程式碼:

if (blah == 2)

FOO(blah);

這將解釋為:

if (blah == 2)

printf("arg is %s", blah);

do_something_useful(blah);;

我們就會發現,if語句只作用於printf(), do_something_useful() 沒按照願意一起執行,即沒有像你預期的那樣被包含在if程式碼中,於是可以使用如下的程式碼塊:

if (blah == 2)

do {

printf("arg is %s", blah);

do_something_useful(blah);

} while (0);

這樣上面的巨集就可以改為:

#define FOO(x) do {

printf("arg is %s", blah);

do_something_useful(blah);

} while (0)

標籤:dowhile0 語言