笔记|C的预处理


预处理知识点

  • 预定义符号
    • 宏定义define
    • 条件编译
    • 文件包含

预定义符号
printf("%s",__FILE__); //进行编译的文件 printf("%d", __LINE__); //文件当前的行号 printf("%s", __DATE__); //文件被编译的日期 printf("%s", __TIME__); //文件被编译的具体时间 //printf("%d", __STDC__); //编译器是否遵顼ASNI C协议,如果定义了则值为1,否则未定义该符号

几种预定义符号极其作用如上,因为这里使用的是vs2013,没有支持ASNI C,所以STDC未定义,在这里注释掉,如果在gcc等编译器下可以显示。
结果如下
笔记|C的预处理
文章图片

宏定义define 格式为:#define XXX ABC
其作用是将XXX完全替换为ABC。
#define MAX 123 printf("%d",MAX);

执行结果如下
笔记|C的预处理
文章图片

为什么说是完全替换,请看如下代码
#define ADD(A,B) A*B printf("%d",ADD(3+2,5));

完全替换为3+2*5而不是惯性思维以为的(3+2)*5
执行结果如下
笔记|C的预处理
文章图片

要想实现(3+2)*5,可以用括号括起来
#define ADD(A,B) ((A)*(B)) printf("%d",ADD(3+2,5));

执行结果如下
笔记|C的预处理
文章图片

注意:
  1. 宏定义define的规范写法为其中的语句尽量大写,小写虽然不报错但是不规范。
  2. 尽量避免在宏定义中使用自增自减运算符,以免产生不必要的麻烦
宏相对于函数:
  1. 宏定义没有参数和函数栈帧建立过程运行速度要比函数快
  2. 函数需要声明类型,宏定义不需要,但是宏定义不够严谨
  3. 宏功能定义无法调试
  4. 宏无法实现一些复杂功
  5. 宏可能会带来运算符优先级问题,容易出错
条件编译 在编译一个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便的。因为我们有条件编译指令。
类似if判断语句,满足条件就编译,不满足条件就不参与编译。
用#if开始,以#endif结束。
#if 1//1为真 printf("%d",666); #endif

笔记|C的预处理
文章图片

当然,有if的地方就有else
#if 0//0为假 printf("%d",666); #else printf("%d", 888); #endif

笔记|C的预处理
文章图片

条件编译语句和宏一样,都是在编译阶段完成了。
defined也可以实现条件编译,其可以判断输入是否定义,如果定义则为真,编译代码。
#if defined(MAX)//MAX未定义,所以不会执行 printf("%d",666); #endif printf("%d", 888);

【笔记|C的预处理】笔记|C的预处理
文章图片

当然,和if-else语句一样,#if和#endif也可以嵌套。
文件包含 通常在编写C/C++代码时,我们会引头文件
头文件的包含方式有两种:
#include #include "stdio.h"

<>引头文件的方式,编译器只会在库里寻找头文件,而以“”方式引的头文件,通常会现在当前目录下寻找头文件,如果未找到,再去库里寻找。
通常在引库里的头文件时,优先选择<>。能免去在目录下搜索头文件的时间。而自己编写的头文件,则会使用“”。也可以使用“”引库里的头文件,此时可以在头文件名字前加上头文件的绝对路径,以增加访问速度。
当然,#include也是预处理指令。此外还有#undef,#pragma等等,这里就不再过多介绍了,有兴趣可以自行查阅。

    推荐阅读