巨集定義中的特殊引數(#、##、...和__VA_ARGS__) 【轉】
阿新 • • 發佈:2019-02-17
轉自:http://blog.csdn.net/cqupt_chen/article/details/8055215
最近在android的某個程式碼的標頭檔案中發現很多__VA_ARGS__,google一下.還是比較有用.附帶其它巨集定義引數,一起記錄之.
1. ...和__VA_ARGS__
看看msdn上給得例子吧
[cpp] view plaincopyprint?- #include <stdio.h>
- #define EMPTY
- #define CHECK1(x, ...) if (!(x)) { printf(__VA_ARGS__); }
- #define CHECK2(x, ...) if ((x)) { printf(__VA_ARGS__); }
- #define CHECK3(...) { printf(__VA_ARGS__); }
- #define MACRO(s, ...) printf(s, __VA_ARGS__)
- int main() {
- CHECK1(0, "here %s %s %s", "are", "some", "varargs1(1)\n");
- CHECK1(1, "here %s %s %s", "are", "some", "varargs1(2)\n"); // won't print
- CHECK2(0, "here %s %s %s", "are", "some", "varargs2(3)\n"
- CHECK2(1, "here %s %s %s", "are", "some", "varargs2(4)\n");
- // always invokes printf in the macro
- CHECK3("here %s %s %s", "are", "some", "varargs3(5)\n");
- MACRO("hello, world\n");
- // MACRO("error\n", EMPTY); would cause C2059
- }
我們看看輸出:
here are some varargs1(1)
here are some varargs2(4)
here are some varargs3(5)
hello, world
是不是很有趣,以後我們如果在一些情況下要定義自己的列印函式就特別方便了
比如我們要在A條件成立,且B條件不成立的情況下列印輸出
可以做如下定義:
[cpp] view plaincopyprint?- #define CHECK1(A,B,...) if ((A)&&!(B)) { printf(__VA_ARGS__); }
記住一點...只能代替後面的引數,如下定義可行不通: [cpp] view plaincopyprint?
- #define W(x,...,y)
是不是很像C++的函式的預設引數一樣.
2. #
這個特殊的巨集定義引數也特別有用.#作為一個預處理運算子,它可以把語言符號字串化(stringizing).例如我們定義的變數等.
看一個簡單的例子:
[cpp] view plaincopyprint?- #include <stdio.h>
- #define TEST(x) printf("square of " #x " is %d.\n",(x)*(x))
- void main()
- {
- int y =4;
- TEST(y);
- TEST(6-3);
- TEST(y+3);
- }
輸出結果:
[cpp] view plaincopyprint?- square of y is 16.
- square of 6-3 is 9.
- square of y+3 is 49.
3. ##
##運算子可以用於類函式巨集的替換部分.##還可以用於類物件巨集的替換部分.這個運算子可以把兩個語言符號組合成單個語言符號.
例如:
[cpp] view plaincopyprint?- #define XNAME(n) x##n
- 執行巨集呼叫:
- int XNAME(4)
- 展開後:
- x4;//相當於直接定義 int x4 這裡也就體現了兩個語言符號的拼接.不過在android經常看到有關##拼接語言符號的情況.