雷火电竞-中国电竞赛事及体育赛事平台

歡迎來(lái)到入門教程網(wǎng)!

C語(yǔ)言

當(dāng)前位置:主頁(yè) > 軟件編程 > C語(yǔ)言 >

c++中do{...}while(0)的意義和用法

來(lái)源:本站原創(chuàng)|時(shí)間:2020-01-10|欄目:C語(yǔ)言|點(diǎn)擊:

linux內(nèi)核和其他一些開源的代碼中,經(jīng)常會(huì)遇到這樣的代碼:

do{
 ...
}while(0)
這樣的代碼一看就不是一個(gè)循環(huán),do..while表面上在這里一點(diǎn)意義都沒有,那么為什么要這么用呢?

實(shí)際上,do{...}while(0)的作用遠(yuǎn)大于美化你的代碼。查了些資料,總結(jié)起來(lái)這樣寫主要有以下幾點(diǎn)好處:

1、輔助定義復(fù)雜的宏,避免引用的時(shí)候出錯(cuò):

舉例來(lái)說(shuō),假設(shè)你需要定義這樣一個(gè)宏:

復(fù)制代碼 代碼如下:

#define DOSOMETHING()\
               foo1();\
               foo2();

這個(gè)宏的本意是,當(dāng)調(diào)用DOSOMETHING()時(shí),函數(shù)foo1()和foo2()都會(huì)被調(diào)用。但是如果你在調(diào)用的時(shí)候這么寫:

復(fù)制代碼 代碼如下:

if(a>0)
    DOSOMETHING();

因?yàn)楹暝陬A(yù)處理的時(shí)候會(huì)直接被展開,你實(shí)際上寫的代碼是這個(gè)樣子的:

復(fù)制代碼 代碼如下:

if(a>0)
    foo1();
foo2();

這就出現(xiàn)了問(wèn)題,因?yàn)闊o(wú)論a是否大于0,foo2()都會(huì)被執(zhí)行,導(dǎo)致程序出錯(cuò)。

那么僅僅使用{}將foo1()和foo2()包起來(lái)行么?

我們?cè)趯懘a的時(shí)候都習(xí)慣在語(yǔ)句右面加上分號(hào),如果在宏中使用{},代碼里就相當(dāng)于這樣寫了:“{...};”,展開后就是這個(gè)樣子:

復(fù)制代碼 代碼如下:

if(a>0)
{
    foo1();
    foo2();
};

這樣甚至不會(huì)編譯通過(guò)。所以,很多人才采用了do{...}while(0);

復(fù)制代碼 代碼如下:

#define DOSOMETHING() \
        do{ \
          foo1();\
          foo2();\
        }while(0)\
...
 
if(a>0)
    DOSOMETHING();
...

這樣,宏被展開后,才會(huì)保留初始的語(yǔ)義。GCC提供了Statement-Expressions用以替代do{...}while(0);

所以你也可以這樣定義宏:

復(fù)制代碼 代碼如下:

#define DOSOMETHING() ({\
        foo1(); \
        foo2(); \
})

2、避免使用goto對(duì)程序流進(jìn)行統(tǒng)一的控制:

有些函數(shù)中,在函數(shù)return之前我們經(jīng)常會(huì)進(jìn)行一些收尾的工作,比如free掉一塊函數(shù)開始malloc的內(nèi)存,goto一直都是一個(gè)比較簡(jiǎn)便的方法:

復(fù)制代碼 代碼如下:

int foo()
{
    somestruct* ptr = malloc(...);
 
    dosomething...;
    if(error)
    {
        goto END;
    }
 
    dosomething...;
    if(error)
    {
        goto END;
    }
    dosomething...;
 
END:
    free(ptr);
    return 0;
 
}

由于goto不符合軟件工程的結(jié)構(gòu)化,而且有可能使得代碼難懂,所以很多人都不倡導(dǎo)使用,那這個(gè)時(shí)候就可以用do{}while(0)來(lái)進(jìn)行統(tǒng)一的管理:

復(fù)制代碼 代碼如下:

int foo()
{
    somestruct* ptr = malloc(...);
 
    do{
        dosomething...;
        if(error)
        {
            break;
        }
 
        dosomething...;
        if(error)
        {
            break;
        }
        dosomething...;
    }while(0);
 
    free(ptr);
    return 0;
 
}

這里將函數(shù)主體使用do()while(0)包含起來(lái),使用break來(lái)代替goto,后續(xù)的處理工作在while之后,就能夠達(dá)到同樣的效果。

3、避免空宏引起的warning

內(nèi)核中由于不同架構(gòu)的限制,很多時(shí)候會(huì)用到空宏,在編譯的時(shí)候,空宏會(huì)給出warning,為了避免這樣的warning,就可以使用do{}while(0)來(lái)定義空宏:

#define EMPTYMICRO do{}while(0)
4、定義一個(gè)單獨(dú)的函數(shù)塊來(lái)實(shí)現(xiàn)復(fù)雜的操作:

當(dāng)你的功能很復(fù)雜,變量很多你又不愿意增加一個(gè)函數(shù)的時(shí)候,使用do{}while(0);,將你的代碼寫在里面,里面可以定義變量而不用考慮變量名會(huì)同函數(shù)之前或者之后的重復(fù)。

上一篇:C語(yǔ)言實(shí)現(xiàn)奇數(shù)階魔方陣的方法

欄    目:C語(yǔ)言

下一篇:C語(yǔ)言求兩個(gè)字符串的最長(zhǎng)公共子串

本文標(biāo)題:c++中do{...}while(0)的意義和用法

本文地址:http://www.jygsgssxh.com/a1/Cyuyan/3165.html

網(wǎng)頁(yè)制作CMS教程網(wǎng)絡(luò)編程軟件編程腳本語(yǔ)言數(shù)據(jù)庫(kù)服務(wù)器

如果侵犯了您的權(quán)利,請(qǐng)與我們聯(lián)系,我們將在24小時(shí)內(nèi)進(jìn)行處理、任何非本站因素導(dǎo)致的法律后果,本站均不負(fù)任何責(zé)任。

聯(lián)系QQ:835971066 | 郵箱:835971066#qq.com(#換成@)

Copyright © 2002-2020 腳本教程網(wǎng) 版權(quán)所有