深入理解C预处理器
C 预处理器不是编译器的组成部分,是编译过程中一个单独的步骤。C预处理器只是一个文本替换工具,它会指示编译器在实际编译之前完成所需的预处理。
所有的预处理器命令都是以井号(#)开头。它必须是第一个非空字符,为了增强可读性,预处理器指令应从第一列开始。
下表包含所有重要的预处理器指令:
指令 |
描述 |
#define |
定义宏 |
#include |
包含一个源代码文件 |
#undef |
取消已定义的宏 |
#ifdef |
如果宏已经定义,则返回真 |
#ifndef |
如果宏没有定义,则返回真 |
#if |
如果给定条件为真,则编译下面代码 |
#else |
#if 的替代方案 |
#elif |
如果前面的 #if 给定条件不为真,当前条件为真,则编译下面代码 |
#endif |
结束一个 #if……#else 条件编译块 |
#error |
当遇到标准错误时,输出错误消息 |
#pragma |
使用标准化方法,向编译器发布特殊的命令到编译器中 |
#line |
提供用于编译器信息的行号 |
预定义宏
ANSI C定义了许多宏。在编程中可以使用这些宏,但是不能直接修改这些预定义的宏。
宏 |
描述 |
__DATE__ |
当前日期,以 "MMM DD YYYY" 格式表示的字符常量。 |
__TIME__ |
当前时间,以 "HH:MM:SS" 格式表示的字符常量。 |
__FILE__ |
包含当前文件名,字符串常量。 |
__LINE__ |
包含当前行号,十进制常量。 |
__STDC__ |
当编译器以ANSI标准编译时,则定义为 1。 |
__STDC__VERSION__ |
如果编译器遵循C99,则该宏的值是199901L,其他情况下,该宏没定义。 |
__STDC__HOSTED__ |
当前是宿主系统,该宏值为1,当前是独立系统,该宏值为0。 |
__STDC__IEC__559__ |
若浮点实现遵循IEC 60599标准,该宏值为1,否则无定义。 |
__STDC__IEC__559__COMPLEX__ |
若复数运算实现遵循IEC 60559标准,则该宏值为1,否则未定义。 |
__STDC__ISO10646__ |
定义为一个长整数常量。 |
预处理器运算符
C预处理器提供了下列运算符帮助创建宏:
1、宏延续运算符(\)
一个宏通常写在一个单行上。但是如果宏太长,单行容纳不下,则使用宏延续运算符(\)。
2、字符串常量化运算符(#)
在宏定义中,当需要把宏的参数转换为字符串常量时,则使用字符串常量化运算符(#)。在宏中使用的该运算符有一个特定的参数或参数列表。
3、标记粘贴运算符(##)
宏定义内的标记粘贴运算符(##)会合并两个参数。它允许在宏定义中两个独立的标记被合并为一个标记。
4、defined()运算符
预处理器defined运算符是用在常量表达式中,用来确定一个标识符是否已经使用 #define定义过。如果指定的标识符已定义,则值为真(非零)。如果指定的标识符未定义,则值为假(零)。
参数化的宏
CPP一个强大的功能是可以使用参数化的宏来模拟函数。
在使用带有参数的宏之前,必须使用#define指令定义。参数列表是括在圆括号内,且必须紧跟在宏名称的后边。宏名称和左圆括号之间不允许有空格。