編寫(xiě)代碼的過(guò)程中,經(jīng)常會(huì)輸出一些調(diào)試信息到屏幕上,一般會(huì)調(diào)用printf這類(lèi)的函數(shù)。
但是當(dāng)調(diào)試解決之后,我們需要手工將這些地方刪除或者注釋掉。
最近在看《Linux C編程一站式學(xué)習(xí)》這本書(shū),就想到一個(gè)方法:
void myprintf(char* fmt, ...)
{
}
#ifdef DEBUG
#define printf(fmt, args...) myprintf(fmt, ##args)
#endif
調(diào)試階段帶著DEBUG調(diào)試,正式上線就可以把printf變成一個(gè)空函數(shù)了。
這樣做的一個(gè)潛在風(fēng)險(xiǎn)是可能會(huì)導(dǎo)致默寫(xiě)glib函數(shù)需要調(diào)用printf輸出錯(cuò)誤log也給取消掉了。
令人欣慰的是,大部分glib調(diào)用的應(yīng)該是fprintf。
雖然問(wèn)題解決了,但是我對(duì)args...以及##args還是不太了解。上網(wǎng)找了些gcc手冊(cè)的資料如下:
帶有可變參數(shù)的宏(Macros with a Variable Number of Arguments)
在1999年版本的ISO C 標(biāo)準(zhǔn)中,宏可以象函數(shù)一樣,定義時(shí)可以帶有可變參數(shù)。宏的語(yǔ)法和函數(shù)的語(yǔ)法類(lèi)似。
下面有個(gè)例子:
#define debug(format, ...) fprintf (stderr, format, __VA_ARGS__)
這里,‘…'指可變參數(shù)。這類(lèi)宏在被調(diào)用時(shí),它(這里指‘…')被表示成零個(gè)或多個(gè)符號(hào),包括里面的逗號(hào),一直到到右括弧結(jié)束為止。當(dāng)被調(diào)用時(shí),在宏體(macro body)中,那些符號(hào)序列集合將代替里面的__VA_ARGS__標(biāo)識(shí)符。更多的信息可以參考CPP手冊(cè)。
GCC始終支持復(fù)雜的宏,它使用一種不同的語(yǔ)法從而可以使你可以給可變參數(shù)一個(gè)名字,如同其它參數(shù)一樣。例如下面的例子:
#define debug(format, args...) fprintf (stderr, format, args)
這和上面舉的那個(gè)ISO C定義的宏例子是完全一樣的,但是這么寫(xiě)可讀性更強(qiáng)并且更容易進(jìn)行描述。
GNU CPP還有兩種更復(fù)雜的宏擴(kuò)展,支持上面兩種格式的定義格式。
在標(biāo)準(zhǔn)C里,你不能省略可變參數(shù),但是你卻可以給它傳遞一個(gè)空的參數(shù)。例如,下面的宏調(diào)用在ISO C里是非法的,因?yàn)樽址竺鏇](méi)有逗號(hào):
debug ("A message")
GNU CPP在這種情況下可以讓你完全的忽略可變參數(shù)。在上面的例子中,編譯器仍然會(huì)有問(wèn)題(complain),因?yàn)楹暾归_(kāi)后,里面的字符串后面會(huì)有個(gè)多余的逗號(hào)。
為了解決這個(gè)問(wèn)題,CPP使用一個(gè)特殊的‘##'操作。
書(shū)寫(xiě)格式為:
#define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)
這里,如果可變參數(shù)被忽略或?yàn)榭眨?#'操作將使預(yù)處理器(preprocessor)去除掉它前面的那個(gè)逗號(hào)。如果你在宏調(diào)用時(shí),確實(shí)提供了一些可變參數(shù),GNU CPP也會(huì)工作正常,它會(huì)把這些可變參數(shù)放到逗號(hào)的后面。象其它的pasted macro參數(shù)一樣,這些參數(shù)不是宏的擴(kuò)展。具體參見(jiàn)《Linux C編程一站式學(xué)習(xí)》,順便夸贊下這本書(shū),寫(xiě)的很好!