我们知道在编译过程中的预处理阶段会进行头文件展开,宏替换以及条件编译。
1.头文件展开:
首先是头文件展开,头文件的前面都有#符号,其实是理指令。
理指令是以#号开头的代码行。#号必须是该行除了任何空白字符外的第一个字符。#后是指令关键字,在关键字和#号
之间允许存在任意个数的空白字符。整行语句构成了一条预处理指令,该指令将在编译器进行编译之前对源代码做某
些转换。下面是部分预处理指令:
# 空指令,无任何效果
#include 包含一个源代码文件
#define 定义宏
#undef 取消已定义的宏
#if 如果给定条件为真,则编译下面代码
#ifdef 如果宏已经定义,则编译下面代码
#ifndef 如果宏没有定义,则编译下面代码
#elif 如果前面的#if给定条件不为真,当前条件为真,则编译下面代码,其实就是else if的简写
#endif 结束一个#if……#else条件编译块
#error 停止编译并显示错误信息
2.宏定义:
宏定义了一个代表特定内容的标识符。预处理过程会把源代码中出现的宏标识符替换成宏定义时的值。宏最常见的用
法是定义代表某个值的全局符号。
宏的第二种用 法是定义带参数的宏(宏函数),这样的宏可以象函数一样被调用,但它是在调用语句处展开宏,并用
调用时的实际参数来代替定义中的形式参数。
像这样#define MAX(a,b) a> b?a:b 就是定义了一个宏函数。
宏函数与自定义函数相比:宏函数可以省去分配和释放栈帧,传参,传返回值等一系列工作,因此那些简短并且被频
繁调用的函数可以用函数式宏定义来代替实现。
但是宏函数发生在预处理阶段,只进行盲目替换,不进行语法检查,所以容易出错,而且不能实现一些复杂的功能。
一些内置宏定义:
__func__ 显示这行代码所在函数的函数名
__LINE__ 显示行号
__TIME__ 显示当前时间
__DATE__ 显示当前日期
3.条件编译:
条件编译我们会经常看到这样的一些指令如:#if #endif #else #elif等等。
我们需要掌握#ifdef和#ifndef指令。
这二者主要用于防止头文件重复包含带来的重复定义。我们一般在.h头文件前面加上这么一段:
//头文件防止重复包含
//funcA.h
#ifndef FUNCA_H
#define FUNCA_H
//头文件内容
#end if
这样,如果a.h包含了funcA.h,b.h包含了a.h、funcA.h,重复包含,会出现一些type redefination之类的错误。
#if defined等价于#ifdef; #if !defined等价于#ifndef