#与##在宏定义中的--宏展开
#與##在宏定義中的--宏展開
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
? printf("%s/n", h(f(1,2))); // 12
? printf("%s/n", g(f(1,2))); // f(1,2)
? return 0;
}
宏展開時:
如果宏定義以#開頭,不展開參數,直接替換。
故g(f(1,2))--->#f(1,2)--->"f(1,2)";
如果宏定義不以#開頭,展開參數,直接替換,由外層向里層,如果碰到的是#開頭的宏,不繼續往里層展開,往外層展開。
由外層向里層,如果碰到的是以非#開頭的宏,繼續往里層走,直至最里層,開始一層層往外層展開。
故h(f(1,2))--->h(12)--->g(12)---->#12----->"12"。
PS:
##在宏中定義,是字符連接符
如a##b##c 等同于 "abc"
#在宏開頭出現,是表示宏展開的方式不同
#a 等同于"a"
#abc 等同于 "abc"
復雜的:
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
? char a = 'a';
? cout<<g(a)<<endl; // a
? cout<<g(g(a))<<endl; // a
? printf("%s/n", h(f(1,2))); // 12
? printf("%s/n", g(f(1,2))); // f(1,2)
? printf("%s/n", g(h(f(1,2)))); // h(f(1,2))
? printf("%s/n", h(g(f(1,2)))); // "f(1,2)"
? printf("%s/n", h(h(f(1,2)))); // "12"
? system("pause");
? return 0;
}
預處理后的:(在編譯選項中添加/EP /P后編譯生成的.i文件)
int main()
{
? char a = 'a';
? cout<<"a"<<endl;
? cout<<"g(a)"<<endl;
? printf("%s/n", "12");
? printf("%s/n", "f(1,2)");
? printf("%s/n", "h(f(1,2))");
? printf("%s/n", "/"f(1,2)/"");
? printf("%s/n", "/"12/"");
? system("pause");
? return 0;
}
---------------------------------------------------
宏解析
1. ##操作符
##操作符它的作用是在替代表中將其前后的參數連接成為一個預處理符號,它不能出現于宏替代表的開端和末尾。
例:
#define concat(s,t) s##t
#define AAA ABC
concat(A, AA)
將被替換成
ABC
2. 重新掃描和替換
在替換列表中的所有參數替換過之后,預處理器將對結果token序列重新掃描以便對其中的宏再次替換。
當正在替換的宏在其替換列表中發現自身時,就不再對其進行替換。今兒,在任何正在嵌套替換的宏的替換過程中遇到正被替換的宏就對其不再進行替換(防止遞歸)。
例:
#define ROOT AAA CCC
#define AAA ROOT
ROOT
將被替換成
ROOT CCC
轉載于:https://www.cnblogs.com/pang123hui/archive/2011/02/25/2309915.html
總結
以上是生活随笔為你收集整理的#与##在宏定义中的--宏展开的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 如果知道一个控件类型的对话框句柄是编辑框
- 下一篇: 你为谁活?