国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C > 正文

淺談do {...} while (0) 在宏定義中的作用

2020-01-26 14:19:23
字體:
來源:轉載
供稿:網友

如果你是一名C程序員,你肯定很熟悉宏,它們非常強大,如果正確使用可以讓你的工作事半功倍。然而,如果你在定義宏時很隨意沒有認真檢查,那么它們可能使你發狂,浪費N多時間。在很多的C程序中,你可能會看到許多看起來不是那么直接的較特殊的宏定義。

下面就是一個例子:

#define __set_task_state(tsk, state_value)   /   do { (tsk)->state = (state_value); } while (0) 

在Linux內核和其它一些著名的C庫中有許多使用do{...}while(0)的宏定義。這種宏的用途是什么?有什么好處?

Google的Robert Love(先前從事Linux內核開發)給我們解答如下:

do{...}while(0)在C中是唯一的構造程序,讓你定義的宏總是以相同的方式工作,這樣不管怎么使用宏(尤其在沒有用大括號包圍調用宏的語句),宏后面的分號也是相同的效果。

這句話聽起來可能有些拗口,其實用一句話概括就是:使用do{...}while(0)構造后的宏定義不會受到大括號、分號等的影響,總是會按你期望的方式調用運行。

例如:

#define foo(x) bar(x); baz(x) 

然后你可能這樣調用:

foo(wolf); 

這將被宏擴展為:

bar(wolf); baz(wolf); 

這的確是我們期望的正確輸出。下面看看如果我們這樣調用:

if (!feral) foo(wolf); 

那么擴展后可能就不是你所期望的結果。上面語句將擴展為:

if (!feral) bar(wolf); baz(wolf); 

顯而易見,這是錯誤的,也是大家經常易犯的錯誤之一。


幾乎在所有的情況下,期望寫多語句宏來達到正確的結果是不可能的。你不能讓宏像函數一樣行為――在沒有do/while(0)的情況下。

如果我們使用do{...}while(0)來重新定義宏,即:

#define foo(x) do { bar(x); baz(x); } while (0) 

現在,該語句功能上等價于前者,do能確保大括號里的邏輯能被執行,而while(0)能確保該邏輯只被執行一次,即與沒有循環時一樣。

對于上面的if語句,將會被擴展為:

if (!feral) do { bar(wolf); baz(wolf); } while (0); 

從語義上講,它與下面的語句是等價的:

if (!feral) {   bar(wolf);   baz(wolf); } 

這里你可能感到迷惑不解了,為什么不用大括號直接把宏包圍起來呢?為什么非得使用do/while(0)邏輯呢?

例如,我們用大括號來定義宏如下:

#define foo(x) { bar(x); baz(x); } 

這對于上面舉的if語句的確能被正確擴展,但是如果我們有下面的語句調用呢:

if (!feral)   foo(wolf); else  bin(wolf); 

宏擴展后將變成:

if (!feral) {   bar(wolf);   baz(wolf); }; else  bin(wolf); 

大家可以看出,這就有語法錯誤了。

總結:Linux和其它代碼庫里的宏都用do/while(0)來包圍執行邏輯,因為它能確保宏的行為總是相同的,而不管在調用代碼中使用了多少分號和大括號。

以上這篇淺談do {...} while (0) 在宏定義中的作用就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 如皋市| 彭山县| 宜宾县| 湘阴县| 普兰店市| 库尔勒市| 岳池县| 吉木萨尔县| 沁水县| 巴南区| 巴林右旗| 贺兰县| 涟水县| 盘山县| 哈尔滨市| 乐清市| 固阳县| 新疆| 昆山市| 日土县| 汉寿县| 饶阳县| 泸定县| 太仆寺旗| 云浮市| 东乌珠穆沁旗| 泰顺县| 普宁市| 金川县| 年辖:市辖区| 井研县| 郁南县| 黄冈市| 乐业县| 武穴市| 白银市| 七台河市| 卓尼县| 牟定县| 社会| 临邑县|