头文件中的变量定义

Met*_*hos 24 c linker include-guards header-files c-preprocessor

我对C和编译过程的基本知识最近变得生疏.我试图找出以下问题的答案,但我无法连接编译,链接和预处理阶段基础知识.快速搜索谷歌也没什么帮助.所以,我决定来到最终的知识来源:)

我知道:不应该在.h文件中定义变量.可以在那里宣布它们.

原因:因为头文件可能包含在多个位置,因此重新定义变量多次(链接器给出错误).

可能的解决方法:在头文件中使用header-guard并在其中定义变量.

它真的是一个解决方案:不.因为标题保护是用于预处理阶段.那就是告诉编译器这部分已经包含在内并且不再包含它.但是我们的多重定义错误来自链接器部分 - 在编译之后很久.

整件事让我对预处理和链接的工作方式感到困惑.如果已经定义了头部保护符号,我认为预处理将不包括代码.在那种情况下,不应该解决变量问题的多重定义吗?

这些预处理指令会在标题保护下重新定义编码过程,但是链接器仍会获得符号的多个定义,会发生什么?

Ric*_*ton 26

我过去曾经使用的一件事(当全局变量流行时):

var.h文件:

...
#ifdef DEFINE_GLOBALS
#define EXTERN
#else
#define EXTERN extern
#endif
EXTERN int global1;
EXTERN int global2;
...
Run Code Online (Sandbox Code Playgroud)

然后在一个 .c文件中(通常是包含main()的文件):

#define DEFINE_GLOBALS
#include "var.h"
Run Code Online (Sandbox Code Playgroud)

其余的源文件通常只包含"var.h".

请注意,DEFINE_GLOBALS不是标头保护,而是允许根据是否定义来声明/定义变量.此技术允许声明/定义的一个副本.


Jay*_*Jay 24

标头防护可以保护您免受单个源文件中的多个包含,而不是来自多个源文件.我想你的问题源于不理解这个概念.

在编译期间,预处理器保护程序不会从此问题中保存.实际上在编译期间,只有一个源文件被编译成一个obj,符号定义没有被解析.但是,如果在链接器尝试解析符号定义时进行链接,则会看到不止一个定义将其标记为错误而感到困惑.


bma*_*ies 10

你有两个.c文件.他们分开编译.每个都包含您的头文件.一旦.每个人都有一个定义.它们在链接时发生冲突.

传统的解决方案是:

#ifdef DEFINE_SOMETHING
int something = 0;
#endif
Run Code Online (Sandbox Code Playgroud)

然后你只在一个 .c文件中#define DEFINE_SOMETHING .


小智 8

标题保护停止在同一翻译单元中多次包含头文件(即在相同的.c源文件中).如果您将文件包含在两个或多个翻译单元中,则它们不起作用.