编译预处理指令
#开头的是编译预处理指令
它们不是C语言的成分,但是C语言程序离不开它们
#define用来定义一个宏
如
#define PI 3.14159
这样PI这个东西就是3.14159的值
在编译过程中,计算机会把代码中出现的PI全部换成3.14159
也可以有如下定义
#define FORMAT "%f\n"
那么,形如
printf(FORMAT,2*PI*3.0);
实际上就是
printf("%f\n",2*PI*3.0);
如果给FORMAT这个部分加上引号
printf("FORMAT",2*PI*3.0);
这样会爆炸的
#define总结
#define <名字><值>
注意没有结尾的分号,因为不是C的语句
名字必须是一个单词,值可以是任何东西
在C语言的编译器开始编译之前,编译预处理程序(cpp)会把程序中的名字替换成值(完全的文本替换)
这就是叫做C语言的宏:
1.如果一个宏的值中有其他的宏的名字,也是会被替换的
2.如果一个宏的值超过一行,最后一行之前的行末需要加
3.宏的值后面出现的注释不会被当做宏的值的一部分
对于2的使用方式是这样:
#define PRT printf("%f",PI);\
printf("%f",PI2);
这样在函数中使用PRT;那么就会执行这2个printf
预定义的宏:
_FILE_ 输出路径(%s)
_LINE_ 输出行号(%d)
_DATE_ 输出日期(%s)
_TIME_ 输出时间(%s)
带参数的宏
宏可以带参数,形如
#define cube(x) ((x)*(x)*(x))
使用方法:
printf("%d\n",(5));
这样计算机在编译的时候就是
printf("%d\n",(5*5*5));
宏定义的误区:
#define RADTODEG1(x) (x * 57.29578);
#define RADTODEG1(x) (x) * 57.29578;
会导致许多错误,在有带参数的宏的时候,一切都要用括号括住,如
#define RADTODEG1(x) ((x * 57.29578));
宏定义加分号可能导致的情况:
#define PRETTY_PRINT(MSG) printf(MSG);
if(n<10)
PRETTY_PRINT("AAA");
else
PRETTY_PRINT("BBB");
如果在宏定义上加了分号,那么对于编译的时候来说,在语句结束后又来了一个分号,相当于加了个空行 这样if和下面的else就连接不上关系了,对于编译接收过来的信息,就只有单单一个if语句
带参数的宏
在大型程序的代码中使用非常普遍
可以非常复杂,如“产生”函数(在#和##这两个运算符的帮助下)
存在中西方文化差异(西方人很喜欢)
部分宏会被inline函数(C++)替代