
1 概述 在工程规模较小,不是很复杂,与硬件结合紧密,要求移植性的时候,可采用宏定义简化编程,增强程序可读性。 当宏作为常量使用时,C程序员习惯在名字中只使用大写字母。但是并没有如何将用于其他目的的宏大写的统一做法。由于宏(特别是带参数的宏)可能是程序中错误的来源,所以一些程序员更喜欢使用大写字母来引起注意。
无参宏的宏名后不带参数,其定义的一般形式为: #define 标识符 字符串 // 不带参数的宏定义#define MAX 10 注意:不要在宏定义中放置任何额外的符号,比如"="或者尾部加";" 使用#define来为常量命名一些优点:
带参数的仍要遵循上述规则,区别只是宏名后面紧跟的圆括号中放置了参数,就像真正的函数那样。 #define <宏名>(<参数列表>) <宏体> 注意参数列表中的参数必须是有效的c标识符,同时以,分隔 算符优先级问题: #define COUNT(M) M*Mint x=5; print(COUNT(x+1)); print(COUNT(++X)); //结果输出:11 和42 而不是函数的输出36 注意:
解决办法:
分号吞噬问题: #define foo(x) bar(x); baz(x)假设这样调用: if (!feral)foo(wolf); 将被宏扩展为: if (!feral)bar(wolf); baz(wolf); ==baz(wolf);==,不在判断条件中,显而易见,这是错误。如果用大括号将其包起来依然会有问题,例如 #define foo(x) { bar(x); baz(x); }if (!feral) foo(wolf); else bin(wolf); 判断语言被扩展成: if (!feral) {bar(wolf); baz(wolf); }>>++;++<< else bin(wolf); ==else==将不会被执行 解决方法:通过==do{…}while(0) #define foo(x) do{ bar(x); baz(x); }while(0)if (!feral) foo(wolf); else bin(wolf); 被扩展成: #define foo(x) do{ bar(x); baz(x); }while(0)if (!feral) do{ bar(x); baz(x); }while(0); else bin(wolf); 注意:使用do{…}while(0)构造后的宏定义不会受到大括号、分号等的影响,总是会按你期望的方式调用运行。
#的作用就是将#后边的宏参数进行字符串的操作,也就是将#后边的参数两边加上一对双引号使其成为字符串。例如a是一个宏的形参,则替换文本中的#a被系统转化为"a",这个转换过程即为字符串化。 #define TEST(param) #paramchar *pStr=TEST(123); printf("pSrt=%s\n",pStr); //输出结果为字符 ”123“
##运算符也可以用在替换文本中,它的作用起到粘合的作用,即将两个宏参数连接为一个数 #define TEST(param1,param2) (param1##param2)int num =TEST(13,59); printf("num=%d\n",num); //输出结果为:num=1359
作用主要是为了方便管理软件中的打印信息。在写代码或DEBUG时通常需要将一些重要参数打印出来,但在软件发行的时候不希望有这些打印,这时就用到可变参数宏了。 # define PR(...) printf(_VA_ARGS_)2 PR("hello world\n"); 3 4 输出结果:hello world 2 一些建议
#define COMDEF_H //头文件内容 #endif
#define MEM_W(x) (*((word *)(x)))
#define MIN(x,y) (((x) < (y)) ? (x) : (y))
#define WORD_HI(xxx) ((byte) ((word)(xxx) >> 8))
((c)>='a' && (c)<='f'))
|
谢谢,宏定义好用。 |
索嘎,,,,学到了 |