#与##在宏定义中的--宏展开
#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
=========以上是赵忠大哥在我帖子里的回复!-=============
下面是我的测试结果
#include<stdio.h>
#include<string.h>
#include<malloc.h>
#define PI 3.14
#define AREA(R) PI*(R)*(R)
/* #表示直接替换为字符串
* ##表示连接两个宏参数
*/
#define T(sth) #sth
#define STR(X) T(love##X) //如果不加T 则会出现未定义的错误
int main(int argc,char **argv)
{
printf("PI=%f\n",PI);
printf("area=%f\n",AREA(10));
printf("TTTT=%s\n",T(NIHAO));
// char *love=STR(111);
char *love;
love = malloc(10);
strncpy(love,STR(111),10);
printf("STR=%s\n",love);
printf("STR=%s\n",STR(11));
return 0;
}
/* 运行的结果:
* [root@bogon c_study]# ./hong宏
* PI=3.140000
* area=314.000000
* TTTT=NIHAO
* STR=love111
* STR=love11*/
注意事项:
在我写程序的时候开始是直接用STR(11)去当右值给别人,gcc提示说love11未定义,如果直接这样写STR(11)运行后会生成love11但是没有任何类型
需要用T()转换下为字符型的才行,这点纠结了一会!