最近面试的时候,发现关于#define的概念经常出现在笔试题中,正好趁这个机会把概念拿出来捋一下,加深一下印象。
#define是C语言中的一种预处理指令,该指令从#开始,直到第一个换行符结束,所以指令长度仅限一行,也可以用反斜杠将定义延续到下一行。
对于#define的理解的话,可以将它理解成文本替换,不管它是一个简单的定义一个常量还是一个类函数宏。
预处理器不会对宏定义的内容做参数类型检查,计算等工作,所以在实际的使用过程中,要注意到宏定义在具体的上下文代码中展开后是否符合预期。
#define ADD 2+2
Int main(void)
{
Int a = ADD;
Printf(“a = %d\r\n”,a);
A = 5*ADD;
Printf(“a = %d\r\n”,a);
}
如果误认为预处理器会对宏定义内容进行计算,那么输出结果应该是
a = 4
a = 20
但是实际上的输出结果是
a = 4
a = 12
因为宏定义会将内容直接替换到代码中,如果代码中的计算存在优先级不同的情况,那么在不加()去改变优先级的情况下,该代码的计算就变成了5*2+2。
在类函数宏里面也是一样,对宏定义的类函数中的参数是不进行类型检查的,这种规则所带来的好处是,对不同类型数据执行相同的处理操作时,只需要用到一个宏定义,而由于函数的参数是与类型相关,即使这些不同类型数据的处理是相同的,也需要定义多个函数。例如
#define ADD(a,b) a+b
用函数则需要根据数据类型来编写不同参数,但是相同功能的函数,用宏定义就会简单很多。由于宏定义是替换的特性,它还能用来做一些函数很难完成的功能,比如将数据类型作为参数传递。
#define MALLOC(n,type) ((type *)malloc(( n ) * sizeof( type )))
函数和宏的选择
这个表格是《C和指针》上总结的函数和#define的区别,感觉总结得很棒
如果一个简洁的功能会被多次调用,那么用宏会比用函数执行速度快很多,因为不存在函数调用过程的开销,当然,也由于宏的一些特性,不进行参数检查,计算,所以也会有额外的一些安全风险,不过在C99后,提供了新的选择——内联函数,后面有空的时候再讲讲内联函数吧。