- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我知道这是一个常见问题,但我仍然无法完全理解它。
在从多个不同的源文件和头文件生成的C或C++程序中,当使用头保护时,每个头文件是否会在整个代码中只包含一次?
之前有人告诉我,一个头文件(带有包含保护)只会在一个翻译单元中包含一次,但在整个代码中会包含多次。这是真的吗?
如果它在整个代码中只包含一次,当一个文件希望包含它并且预处理器检测到它已经被包含时,希望使用它的那个文件如何知道它在之前包含的代码中的下落?
最佳答案
这是过程:
source header source header header
\ / \ | / /
\ / \ | / /
PREPROCESSOR PREPROCESSOR
| |
V V
preprocessed code preprocessed code
| |
COMPILER COMPILER
| |
V V
object code object code
\ /
\ /
\ /
LINKER
|
V
executable
预处理
#include
是第一步。它指示预处理器处理指定的文件,并将结果插入到输出中。
如果A
包含B
和C
,且B
包含C
, A
的预处理器输出将包含两次 C
的处理文本。
这是一个问题,因为它会导致重复声明。一种补救方法是使用预处理器变量跟踪是否包含源代码(也称为 header 保护)。
#ifndef EXAMPLE_H
#define EXAMPLE_H
// header contents
#endif
第一次,EXAMPLE_H
是未定义的,预处理器将评估 ifndef
/endif
block 中的内容。第二次,它将跳过该 block 。所以处理后的输出变化,定义只包含一次。
这很常见,以至于一些编译器实现了一个非标准指令,它更短并且不需要选择唯一的预处理器变量:
#pragma once
// header contents
您可以确定您希望 C/C++ 代码的可移植性,以及要使用的 header 保护。
header 守卫将确保每个 header 文件的内容在翻译单元的预处理代码中最多出现一次。
编译
编译器从您预处理的 C/C++ 生成机器代码。
通常,头文件仅包含声明,而不包含实际定义(也称为实现)。编译器为当前缺少定义的任何内容都包含一个符号表。
链接
链接器组合目标文件。它将定义(又名实现)与对符号表的引用进行匹配。
可能是两个目标文件提供了定义,链接器会取一个。如果您将可执行代码放在 header 中,就会发生这种情况。这在 C 中通常不会发生,但在 C++ 中却经常发生,因为模板。
header “代码”,无论是声明还是定义,都包含在所有目标文件中多次,但链接器将所有这些合并在一起,因此它仅在可执行文件中出现一次。(我不包括多次出现的内联函数。)
关于c++ - 头文件在整个程序中只包含一次?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30873206/
我有一位客户希望她的网站适合 iPad/iPhone。该网站有一个用 Flash 制作的菜单标题。我使用 Google 的 Swiffy 将文件转换为 flash_1.html 文件。 当网站检测到
我注意到 boost 库使用了 (.hpp) 的头文件。 我很好奇,因为我看到的大多数源文件都使用普通的 .h 头文件。 是否有任何特殊情况需要使用 .hpp 而不是 .h ? 谢谢 最佳答案 只是约
C需要不同的头文件,如stdio.h、stdlib.h、fcntl.h等,对于不同的函数和结构定义,这是为什么?在幕后,一切都归结为 libc 那么为什么不创建一个包含所有定义和原型(prototyp
我是一名工科学生,不是一个非常强大的程序员。我的一项作业包括使用 openGL 创建 VR 程序。我得到了一个使用 gmtl 的模板(我真的不想重写)标题广泛。该作业需要实现一些 sixense模拟中
我正在尝试编译涉及C和C++文件的应用程序。使用一个特定的 header ,我遇到了问题。有问题的文件(C++头文件)如下所示: #ifndef TASK_H #define TASK_H #incl
我编写了我的项目,将main和c源代码保存在一个文件中,并将头文件保存在codeblocks的include目录中。当我从项目主函数中调用我的函数时,它编译得很好......但是当我将 header
我正在尝试将 .so 库导入到 python 代码中以使用 c 函数。我认为使用 from ctypes import * import ctypes lib = CDLL('./libcaenhvw
我有 2 个重叠的头文件,如下所示: header1.h ... __declspec(dllexport) void abc(); __declspec(dllexport) void xyz(
我有一个大型 Java 库,我想开发几个与该库接口(interface)的较小应用程序。该库将作为 JAR 出现在目标设备的类路径中,但我想尽可能避免在编译时出现整个库(JAR 或源代码)。 (如果重
我必须在我的项目中使用相机制造商提供的库。我正在使用 Visual Studio 2015。我在编写#include 后合并了VS 2015 建议的所有头文件。 我已经在VS 2015中指定了包含文件
我在编写我的项目时遇到了这个问题。我有 2 个 header ,每个 header 都有一个类,它们需要另一个,如下所示。 我以为这只是需要使用前向声明,但仍然不起作用。我没有想法。 寻求帮助:D H
我在一个项目中工作,我想在C++中做反射,所以经过研究我发现最好的方法是解析头文件以获得XML格式的抽象语法树并在反射中使用它。我尝试了很多工具,但没有一个与 visual c++ 2008 或 vi
我尝试从 BufferedImage 制作一个 BMP 文件。这是我尝试在 bmp 文件中写入标题和像素的函数。 我有一个错误,但我找不到那个。我需要你的帮助。 static void writeTo
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: undefined reference to `WinMain@16' 我一直在研究循环双链表。决定创建一个
处理这种情况的最佳做法是什么? class A { private: std::vector derp; public: struct B { ...
上下文:我正在为嵌入式板构建一些代码。它需要安装 Xilinx 工具、Linaro 工具链,然后调用开发板构建目录中的设置 bash 脚本(我们称之为 setup.sh)。 如果我不运行 setup.
我尝试使用头文件和源文件,但遇到了问题。因此,我对我正在尝试做的事情做了一个简化版本,我在 CodeBlocks 中遇到了同样的错误(undefined reference to add(double
当我包含用于将某些程序的整数类型转换为字符串类型的#include 头文件时,我的编译器(GCC for C++)抛出错误。谁能帮我解决这个问题? 这是一个C++的小代码,我是第一次尝试。 #incl
我的头文件中有一些错误,我不知道如何修复,因为我是 C++ 的新手。 这是头文件的代码: #pragma once typedef unsigned int uint; class DCEncrypt
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我是一名优秀的程序员,十分优秀!