预定义符号
_FILE_ //进行编译的源文件
_LINE_ //文件当前的行号
_DATE_ //文件被编译的日期
_TIME_ //文件被编译的时间
_STDC_ //如果编译器遵循ANSI C,其值为1,否则未定义
//这些预定义符号都是语言内置的
#define 定义标识符
语法:#define name stuff
#define 定义宏
#define机制包含了一个规定,允许吧参数替换到文本中,这种实现通常称为宏或定义宏。
语法:#define name(parament-list) stuff 其中的parament-list 是一个由逗号隔开的符号表,它们可能出现在stuff中。
注意:参数列表的左括号必须与name紧邻。如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分。
用于对数值表达式进行求值的宏定义应该加上括号,避免在使用宏定义时由于参数中的操作符或邻近操作符之间出现不可预料的相互作用。
#define 替换规则
在程序中扩展#define定义符号和宏时,需要注意以下几个步骤。
- 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号,如果有,它们首先被替换。
- 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。
- 最后,再次对结果文件进行扫描,看看他是否包含任何由#define定义的符号,如果是,就重复上述处理的过程。
注意:
- 宏参数和#define定义中可以出现其他#define定义的变量。但是对于宏,不能出现递归。
- 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。
如何把参数插入到字符串中?
首先我们以下这个代码:
char*p="hello""bit\n";
printf("hello","bit\n");
printf("%s",p);
这里输出的是:hello bit
我们可以发现字符串是有自动连接的特点的
因此 我们可以这样写
#define PRINT(FORMAT,VALUE)
printf("the value is "FORMAT"\n",value);
PRINT("%d",10);
//这里只有当字符串作为宏参数的时候才可以把字符串放在字符串中
另一个方法是,使用#,把一个宏参数变成对应的字符串。比如
int i=10;
#define PRINT(FORMAT,VALUE)
printf("the value of "#value" is "FORMAT"\n",value);
PRINT("%d",i+3);
代码中的#value会预处理器处理为:“VALUE”
所以最终的输出结果应该是:the value of i+3 is13
要与#区别的是##
##的作用是可以把位于他两边的符号合成一个符号。它允许宏定义从分离的文本片段创建标识符。
#define ADD_TO_SUM(sum,value)
sum##num+=value;
ADD_TO_SUM(5,10);//作用是:给sum 5 增加10
注意:这样的连接必须产生一个合法的标识符。否则其结果就是未定义的。