1. 明示常量#define
使用#define定义明示常量,例如:
#define TWO 2 //可以使用注释
#define PI 3.1415923 //定义PI
#define SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL //定义一年有多少秒(不考虑闰年)
上述宏被称为“类对象宏”,其定义的语法如下:
#define PI 3.1415926
//预处理指令 宏 替换体
注意:
- 类似定义一年有多少秒的宏定义,预处理器将为我们计算常量表达式的值,因此我们可以(其实是最好)写出计算过程,而不是写出计算的结果的值。
- 另一方面,#define SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL 将使16位机的整型数溢出,因此要用到长整型L;除此之外,这里用无符号长整型(UL)更加合适。
2. 在#define中使用参数
在#define中使用参数可以创建外形和作用与函数类似的类函数宏。其定义的语法如下:
#define MEAN(X,Y) (((X)+(Y))/2)
// 宏 宏参数 替换体
函数调用时将值传递给函数,宏调用在编译之前将参数记号传递给程序。由于传递的是记号,而不是值,所以在宏定义的时候,要注意加括号,这样遇到传入参数是一个表达式的时候才不会出现错误。
比如下面的例子:
#define SQUARE(X) X*X
这个宏定义当调用SQUARE(X+2)的时候,显然会出现错误。
所以使用带参数的宏定义的时候,一定要注意参数的括号。下面是常用的求两个数较小的那个数的宏定义。使用这个宏定义的时候,需要注意当传入参数带有自增或者自减运算符的时候,用?:这样的三目运算符会造意想不到的结果。
#define MIN(X,Y) ((X)<=(Y)?(X):(Y))
比如:
int least=MIN((*p)++,a);
如果程序的本意得到(*p)++与a中较小的那个值,但是如果(*p)++<=a,那么三目运算符返回(*p)++,此时会改变(*p)的值。而如果传入(++(*p)),则最后会导致least的值也不正确。