宏定義替換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__)