繁体   English   中英

C 预处理器包含指令

[英]C Preprocessor include directive

当我包含另一个源(即stdio.h )时,预处理器是否足够智能以仅包含我在代码中使用的函数?

示例:假设这个小程序只包含我正在使用的内容,以及 printf 函数使用的内容,递归地包含它们,但是更大的程序呢?

#include <stdio.h> 

int main(void) {
   printf("Hello World\n"); 
   return 0;
}

不,恰恰相反:

#include执行文本替换:它打开文件并将其所有内容1复制到您的主 C 文件中。 在此过程中,它执行包含文件中的所有预处理器指令。 除其他事项外,这意味着它将递归地包含 header 中#include d 的所有文件。

#include不知道也不关心你最终使用了包含文件的哪一部分。


1如前所述,预处理器指令在包含的文件执行。 这可以修改包含的内容。 例如,假设以下 header 文件header.h

#ifndef HEADER_H
#define HEADER_H

#ifdef NDEBUG
#  define LOG(...) ((void) 0)
#else
#  define LOG(...) log_message(__FILE__, __LINE__, __VA_ARGS__)

inline void log_message(const char* filename, int line, ...) {
    // Logging code omitted for brevity.
}
#endif

// other stuff

#endif

现在,如果您的main.c文件如下所示:

#define NDEBUG
#include "header.h"

int main(void) {
    // …
    LOG("hello");
}

……然后,在预处理之后,您的main.c文件将如下所示(我省略了一些不相关的内容):

# 1 "main.c"

# 1 "./header.h" 1




# 13 "./header.h"

// other stuff


# 3 "main.c" 2

int main(void) {
    // …
    ((void) 0);
}

…换句话说,只包含了header.h中对应于#ifdef NDEBUG的部分,而不是#else子句中的部分。 如果我们在没有定义NDEBUG的情况下包含header.h ,那么包含的 header 代码将包含log_message的定义。

正如其他人所说, #include将逐字粘贴您要定位的整个文件 但是,您通常会包含标头,这些标头看起来像

extern int a (int b);

extern char * c (void);

static inline int d (int e, int f) {
    ...
}

extern void * g (void * h);

...

上面的代码恰好占用零 memory(除非您开始使用其中一个inline函数),因为它完全由编译器指令组成,没有其他任何内容。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM