预处理:编译之前的处理。
预处理主要的3个方面内容:
(1)宏定义和宏替换
(2)文件包含
(3)条件编译
宏
宏替换的本质------文本替换
(1)宏名一般大写,宏名和参数的空格间不能有空格,宏定义末尾不加分号。如: #define BASE 4
(2)宏只做替换,不做语法检查,不做计算,不做表达式求解
#define FUNC(x) x*x
int a = 3,b = 4;
FUNC(a+b) = a+b*a+b = 3+4*3+4 = 19; //可以发现此处是直接做了文本替换,不会做表达式 a+b 的求解
#define FUNC(x) (x)*(x)
int a = 3,b = 4;
FUNC(a+b) = (a+b)*(a+b) = (3+4)*(3+4) = 49;
(3)宏替换在编译前进行,不分配内存;函数调用在编译后程序运行时进行,并分配内存
(4)函数只有一个返回值,但是利用宏可以设法得到多个值
(5)宏替换使源程序变长,函数调用不会
(6)宏替换不占用运行时间,只占编译时间,函数调用占运行时间(分配内存,保留现场,值传递,返回值)
文件包含
#include <xxx>
:头文件名在尖括号里,可以认为该头文件是标准头文件
。编译器会在预定义的位置集合(可以通过设置查找路径环境变量,或者通过命令行选项修改)中查找该头文件。
#include "xxx.h"
:头文件名在引号里,那么它是非系统头文件
,非系统头文件(自定义的一些文件)的查找通常开始于源文件所在的路径。
条件编译
#if/#ifdef/#ifndef
#elif
#else
#endif
文件包含需要考虑 避免多重包含的问题,可以通过条件编译语句 来避免多重包含。
如下:使用#ifndef
判断 指定的预处理器变量
TESTMGR_H 是否 有被定义,如果 预处理器变量没有被定义,则使用#define
去定义,直到出现#endif
之间的语句都被处理。可以通过这个方法来解决多次包含同一个头文件的情况,这样同一个头文件只会被引入1次。因为 通过 #ifndef 判断该文件已经存在,则不会执行下面的内容。
#ifndef TESTMGR_H
#define TESTMGR_H
// 类的定义与相关函数的定义
#endif
宏连接符 ‘##’ 和 ‘#’
#
的功能是将后面的宏参数进行字符串化操作,即对它所引用的宏变量,替换后在在左右加上一个双引号
‘##’ 连接符