找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3904|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

復(fù)雜的宏定義---之一

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:104126 發(fā)表于 2016-1-23 00:50 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
關(guān)于宏定義我想通過__LINE__表示不同的行號來定義不同的系統(tǒng)時間變量用以不同地方的打印時間,如

關(guān)于宏定義
我想通過__LINE__表示不同的行號來定義不同的系統(tǒng)時間變量用以不同地方的打印時間,如下:

  C/C++code
   #defineLINE__ __LINE__ #define LINE(sys) sys##LINE__ SYSTEMTIME LINE(sys);GetLocalTime(&LINE(sys)); printf( "M/d/d d:d:d.d星期\n",LINE(sys).wYear,LINE(sys).wMonth,LINE(sys).wDay,LINE(sys).wHour,LINE(sys).wMinute,LINE(sys).wSecond,LINE(sys).wMilliseconds,LINE(sys).wDayOfWeek);#endif

編譯發(fā)現(xiàn)__LINE__并不會被先替換成行號,而且多出定義會出現(xiàn)重復(fù)定義的錯誤,直接用#define LINE(sys)sys##__LINE__也會出錯,更直接的在定義的時候:SYSTEMTIMEsys##__LINE__;來定義,錯誤更離譜。這是為什么呢?




1條回答


#與##在宏定義中的--宏展開
#include
#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;
}
宏展開時:
如果宏定義以#開頭,不展開參數(shù),直接替換。
故g(f(1,2))--->#f(1,2)---$amp;>amp;$quot;f(1,2)";
如果宏定義不以#開頭,展開參數(shù),直接替換,由外層向里層,如果碰到的是#開頭的宏,不繼續(xù)往里層展開,往外層展開。
由外層向里層,如果碰到的是以非#開頭的宏,繼續(xù)往里層走,直至最里層,開始一層層往外層展開。
故h(f(1,2))--->h(12)--->g(12)---->#12-----$amp;>amp;$quot;12"。
PS:
##在宏中定義,是字符連接符
如a##b##c 等同于 "abc"
#在宏開頭出現(xiàn),是表示宏展開的方式不同
#a 等同于"a"
#abc 等同于 "abc"
復(fù)雜的:
#include
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
char a = 'a';
cout$amp;
cout$amp;
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;
}
預(yù)處理后的:(在編譯選項中添加/EP /P后編譯生成的.i文件)
int main()
{
char a = 'a';
cout$amp;
cout$amp;
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. ##操作符
##操作符它的作用是在替代表中將其前后的參數(shù)連接成為一個預(yù)處理符號,它不能出現(xiàn)于宏替代表的開端和末尾。
例:
#define concat(s,t) s##t
#define AAA ABC
concat(A, AA)
將被替換成
ABC
2. 重新掃描和替換
在替換列表中的所有參數(shù)替換過之后,預(yù)處理器將對結(jié)果token序列重新掃描以便對其中的宏再次替換。
當正在替換的宏在其替換列表中發(fā)現(xiàn)自身時,就不再對其進行替換。今兒,在任何正在嵌套替換的宏的替換過程中遇到正被替換的宏就對其不再進行替換(防止遞歸)。
例:
#define ROOT AAA CCC
#define AAA ROOT
ROOT
將被替換成
ROOT CCC







分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復(fù) 返回頂部 返回列表