c/c++的printf和宏定義

宏定義替換printf自定義輸出到終端和日誌文件

#define PRT(...) printf(##__VA_ARGS__);sprintf(buf,##__VA_ARGS__);log(buf);

一、#用來把參數轉換成字符串

例子一:

 

#define P(A) printf("%s:%d\n",#A,A);

int main(int argc, char **argv)

{

        int a = 1, b = 2;

        P(a);

        P(b);

        P(a+b);

        system("pause");

}

輸出爲:

 

例子二:

#define SQUARE(x) printf("The square of "#x" is %d.\n", ((x)*(x)));

 

SQUARE(8)

輸出的是:The square of 8 is 64

 

二、##運算符可以用於宏函數的替換部分。這個運算符把兩個語言符號組合成單個語言符號,爲宏擴展提供了一種連接實際變元的手段

 

例子一:

#define XNAME(n) x ## n

 

如果這樣使用宏:XNAME(8)

則會被展開成這樣:x8

 

##就是個粘合劑,將前後兩部分粘合起來,也就是有“字符化”的意思。但是“##”不能隨意粘合任意字符,必須是合法的C語言標示符。在單一的宏定義中,最多可以出現一次“#”或“##”預處理操作符。如果沒有指定與“#”或“##”預處理操作符相關的計算次序,則會產生問題。爲避免該問題,在單一的宏定義中只能使用其中一種操作符(即,一份“#”或一個“##”,或都不用)。除非非常有必要,否則儘量不要使用“#”和“##”。

 

三、__VA_ARGS__ 是一個可變參數的宏,很少人知道這個宏,這個可變參數的宏是新的C99規範中新增的,目前似乎只有gcc支持(VC6.0的編譯器不支持)。
實現思想就是宏定義中參數列表的最後一個參數爲省略號(也就是三個點)。

 

四、##__VA_ARGS__ 宏前面加上##的作用在於,當可變參數的個數爲0時,這裏的##起到把前面多餘的","去掉的作用,否則會編譯出錯

一般這個用在調試信息上多一點

例如:

 

#define my_print1(...)  printf(__VA_ARGS__)   my_print1("i=%d,j=%d\n",i,j)  正確打印

#define my_print2(fmt,...)  printf(fmt,__VA_ARGS__)  

my_print1("i=%d,j=%d\n",i,j) 正確打印

my_print1("iiiiiii\n")       編譯失敗打印,因爲擴展出來只有一個參數,至少要兩個及以上參數

如果是#define my_print2(fmt,...)  printf(fmt,##__VA_ARGS__)  

那麼

my_print1裏面不管是幾個參數都能正確打印

 

#define MODULE_NAME "MY_LIBS"
 
#define error_printf(fmt,...) printf("[ERROR]["MODULE_NAME"](%s|%d)"fmt,__func__,__LINE__,##__VA_ARGS__)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章