引言:
学C语言之初。一提到预处理,脑子里想到的就是#define的宏定义以及#include包括的头文件。后来随着对C的深入学习发现。预处理不止这些。比方条件编译、提前定义的宏等等。以下对此进行总结。
先给出预处理的定义:在编译程序之前,先由预处理器检查程序(因此称为预处理器),依据程序中使用的预处理器命令,预处理器用符号缩略语所代表的内容替换程序中的缩略语。
1. #define
最经常使用的预处理器命令就是define命令,该预处理器命令有三部分组成:#define本身、符号缩略语、替换列表(或称为主体)。
结构可写为:#define 宏 主体 当中宏部分不能有空格,仅仅能是数字、下划线、字母。但第一个字符不能使数字。
如:#define PX printf("x is %d.\n", x)
预处理器发现程序中的宏后,会用它的等价替换文件取代宏。假设该字符串中还包含宏。则继续替换这些宏。假设宏出如今双引號中,则不做替换。
假设在双引號中出现的宏前面有“#运算符”,则能够用宏參数创建字符串。如:
#define PSQR(X) printf("The square of X is %d.\n", ((X)*(X)))
PSQR(8) //等价于printf("The square of X is %d.\n", ((8)*(8)))。输出The square of X is 64.
#define PSQR(X) printf("The square of "#X" is %d.\n", ((X)*(X)))
PSQR(8) //等价于printf("The square of8 is %d.\n", ((8)*(8))),输出The square of8 is 64.
小结:引號中的字符中中的X被看作普通文本,而不是被看作一个可替换的语言符号。#符号用作一个预处理运算符,它能够把语言符号转化为字符串。
预处理器的粘合剂:##运算符。和上面的#运算符一样。##运算符能够用于类函数宏的替换部分。
##还能用于类对象宏的替换部分,把两个符号组合成单个语言符号,如:
#define XNAME(n) x##n
XNAME(4) //等价于x4
从宏的定义和使用能够看出,它与函数很相近,但宏与函数又有所不同,他们的选择实际上是时间和空间的权衡。
宏产生内联代码,也就是说在程序中产生语句。
假设使用宏20次,则会把20行代码插入程序中。假设使用函数20次,那么程序中仅仅有一份函数的语句拷贝,因此节省了空间。
还有一方面。程序的控制必须转移到函数中并随后返回调用程序。因此这比内联代码话费的时间多。
2.#include
预处理器发现#include指令后,就会寻找后跟的文件名称并把这个文件的内容包括到当前文件里。被包括文件的文本将替换源码文件里的#include指令。就像把被保护文件里的所有内容键入到源文件里的这个特定位置一样。
#include <name.h> // 文件放在尖括号里,搜索系统工作文件夹
#include “name.h” // 文件名称放在双引號中,搜索当前工作文件夹
#include "/dir1/dir2/name.h" // 搜索/dir1/dir2文件夹
头文件里一般包括的内容有:明显常量、宏函数、函数声明、结构体模板定义、类型定义
3. 其它指令
#undef指令取消前面的#define定义。
#if #ifdef #ifndef #else #elif #endif指令可用于选择什么情况下编译哪些代码。#line指令用于重置行和文件信息,#error指令用于给出错误消息。#param指令用于想编译器发出指示
4. 提前定义的宏:
__DATE__ : 进行预处理的日期
__FILE__ :代码当前源码文件名称的字符串文字
__LINE__ :代表当前源码文件里的行号的整数常量
__STDC__ :设置为1时。表示该实现遵循C标准
__TIME__ : 源文件编译时间