1. 预处理指令
在第一篇文章中,我们就提到了预处理器的概念。预处理器就是用来处理预处理指令。预处理指令包含以下三种:
A. 宏定义 #define
B. 文件包含 #include
C. 条件编译 #if #ifdef等等。
2. 简单的宏定义
在之前我们就广泛地运用了宏定义,在此不在赘述。
3. 带参数的宏定义
带参数的宏定义,也称为函数式宏。
举个简单的例子:
#define MAX(m,n) ((m)>(n)?(m):(n)) int main (void) { printf("%d",MAX(1,2)); return 0; }
相信这个例子看完大家都明白了,在此只说明一点:在程序中,出现了很多(),那是因为为了防止替换时的副作用,因为预处理器是不懂C语言语法规则的,一般来说只是进行简单的机械替换,因此如果m,n为表达式,很可能出现问题。
例如:
#define MUL(m,n) m*n int main (void) { printf("%d",MUL(1+3,3+1)); return 0; }
这个程序就是有问题的,大家可以自己测试。
那么我们为什么需要带参数的宏呢?优点有两个:
A. 效率更高。如果定义成一般的函数,程序在执行时就需要存储上下文信息,把参数压栈等等,但是宏定义显然没有这样的运行开销。(这里我们可以想一下内联函数。)
B. 宏没有类型限制。其实就是参数类型重载,例如在上面的MAX定义中,可以传入int类型的参数,也可以传入double类型的参数。
那么缺点我个人认为最明显的一点也就是会造成程序代码的膨胀。
4. 一些琐碎
#运算符可以把宏定义的一个参数转换为字符串字面量,举个例子:
#define PRINT(num) (printf(#num "=%d",num)) int main (void) { PRINT(10/2); return 0; }大家可以自己运行看下结果。
#undef指令,取消宏定义。