18.1 C 预处理器
顾名思义,预处理器在编译器运行之前在源文件上运行。它的能力非常有限,但你可以在构建 C 程序时使用它们来发挥巨大的优势。
预处理器会生成一个新文件,这是编译器实际处理的文件。预处理器的所有命令都从行首开始,# 符号作为第一个非空白字符。
预处理器的主要用途是在源文件中执行文本替换,但它也会使用 #if
或类似的语句来执行一些基本的条件代码。
让我们从最常见的预处理器指令开始:#include
。
#include
#include
用于将一个文件的内容拉取到当前源文件中。#include
没有什么复杂的内容。它从文件系统中读取文件,在该文件上运行预处理器,并将结果放入输出文件中。这是为每个 #include
指令递归达成的。
例如,如果你查看 Modules/_multiprocessing/semaphore.c
文件,那么在文件顶部附近,你将看到以下行:
此行代码会告诉预处理器获取到 multi-processing.h
的全部内容,并将它们放入到输出文件的同个位置上。
你需要注意到 #include
语句有两种不同形式。其中一个使用引号("")指定所包含的文件名,另一个使用括号(<>)。两者区别在于在文件系统上查找文件时的搜索路径。
如果你使用 <>
作为文件名,则预处理器将仅查看系统包含文件。在文件名周围使用引号将强制预处理器首先查找本地目录,然后再回退到系统目录。
#define
#define 允许你执行简单的文本替换,还可以使用 #if 指令,你将会在下面看到。
在最基本的情况下,#define 允许你定义一个新的符号,该符号将在预处理器输出中替换文本字符串。
仍然是在 semphore.c
中,你会发现这样的一行文本:
此行文本告诉预处理器在将代码发送到编译器之前,用文本字符串 NULL
替换低于此行的 SEM_FAILED
的所有实例。
#define
定义项也可以在 Windows
特定版本的 SEM_CREATE
中使用参数:
在这种情况下,预处理器将期望 SEM_CREATE()
看起来像一个函数调用,并有三个参数。这通常被称为宏。它直接将这三个参数的文本替换到输出代码中。
例如,在 semphore.c
的第 460 行,SEM_CREATE
宏是这样使用的:
当你为 Windows 编译时,此宏将被展开,使此行看起来像这样:
在后面的章节中,你将看到此宏在 Windows 和其他操作系统上的定义有什么不同。
#undef
此指令将从 #define
中删除任何以前的预处理器定义。这使得 #define
仅对文件中的部分有效成为可能。
#if
预处理器还允许使用条件语句,允许你根据某些条件包括或排除文本部分。使用 #endif
指令来关闭条件语句,还可以使用 #elif
和 #else
进行微调。
你将在 CPython 源代码中看到 #if
的三种基本形式:
如果定义了指定的宏,则
#ifdef <macro>
会包含后面的文本块。你还可以看到它被写成#if defined(<macro>)
。如果没有定义指定的宏,则
#ifndef <macro>
会包含后面的文本块。如果定义了指定的宏并且求得的值为
True
,则#if <macro>
会包含后面的文本块。
这里要注意的是,使用“文本”而不是“代码”来描述文件中所包含或排除的内容,是因为预处理器对 C 语法一无所知,也不关心指定的文本是什么。
#pragma
Pragma 是编译器的指令或提示。一般来说,你可以在阅读代码时忽略这些代码,因为它们通常用于处理代码是如何编译的,而不是代码是如何运行的。
#error
最后,#error
会显示一条消息,并导致预处理器停止执行。同样,你在阅读 CPython 源代码时可以安全地忽略这些内容。
Last updated