#define定义宏
- 定义:
#define机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro).
- 声明方式:
#define name(parament-list) stuff
其中的 parament-list 是一个由逗号隔开的符号表,它们可能出现在stuff中。
注:参数的左括号必须与name紧邻,如果两者之间有任何的空白存在,参数列表会被解释为 stuff的一部分。
程序编译时宏被#define替换:
1.在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号,如果是,首先被替换。
2.替换文本随后插到程序中原来文本的位置,对于宏,函数名被他们的值替换。
3.最后,再次对结果文件进行扫描,看看他是否包含任何#define定义的符号,如果是,复述上述过程。注:
(1)宏参数和#define定义中可以出现其它的#define定义变量,但是对于宏不能出现递归。
(2)当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。
带副作用的宏参数
当宏参数在宏的定义超过一次时,如果参数带有副作用,那么你使用的这个宏很有可能出现危险,导致不可预测的后果,副作用就是表达式求值的时候出现的永久性的效果。
例:
x+1;//不带副作用
x++;//带有负作用
MAX宏可以证明具有副作用的参数引起的问题:
#define MAX(a, b) ((a) > (b) ? (a) : (b))
......
x = 5;
y = 8;
z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);//输出的结果是什么?
预编译处理后的结果:
z = ((x++) > (y++) ? (x++) : (y++));
输出结果是:x=6 y=10 z=9
命名约定
一般来讲函数和宏的使用语法相似,所以语言本身没法帮助我们区分二者,因此习惯:
(1)把宏名全部大写。
(2)函数名不要全部大写。
宏和函数的区别:
属性 | define宏 | 函数 |
---|---|---|
代码长度 | 每次使用时,宏代码都是被插入到程序中去的,除非非常小的宏之外,程序的长度将大幅增长。 | 函数代码只出现一个地方,每次使用这个函数时,都调用那个地方的同一份代码。 |
执行速度 | 更快 | 存在函数调用/返回的额外开销。 |
操作符优先级 | 宏参数的求值是在所有周围表达式的上下文环境里,除非它们加上括号,否则临近操作符的优先级可能会产生不可预料的结果。 | 函数参数只在函数调用时求值一次,它的结果值传递给函数,表达式的求值结果更容易预测。 |
参数求值 | 参数每次用于宏定义时,它们都将重新求值,由于多次求值,具有副作用的参数可能产生不可预料的结果。 | 参数在函数调用前只求值一次,在函数中多次使用参数并不会导致多种求值过程,不会造成任何特殊问题。 |
参数类型 | 宏宇类型无关,只要对参数的操作是合法的,它可以使用任何参数类型。 | 函数的参数是与类型有关的,如果参数的类型不同就需要不同的函数,即使它们执行的任务是相同的。 |