- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
如果我有一个带有结构的 source.c 文件:
struct a {
int i;
struct b {
int j;
}
};
如何在另一个文件(即 func.c
)中使用该结构?
我应该创建一个新的头文件,在那里声明结构并将该头文件包含在 func.c
中吗?
或者我应该在头文件中定义整个结构并将其包含在 source.c
和 func.c
中吗?如何在两个文件中都将结构声明为 extern
?
我应该 typedef
吗?如果是,怎么办?
最佳答案
当一个类型在文件中使用时(即 func.c 文件),它必须是可见的。最糟糕的做法是将其复制粘贴到每个需要它的源文件中。
正确的做法是放在头文件中,需要的时候include这个头文件。
这是我比较喜欢的方案,因为它让代码高度模块化。我会将您的结构编码为:
#ifndef SOME_HEADER_GUARD_WITH_UNIQUE_NAME
#define SOME_HEADER_GUARD_WITH_UNIQUE_NAME
struct a
{
int i;
struct b
{
int j;
}
};
#endif
我会将使用此结构的函数放在同一个标头中(该函数是其“接口(interface)”的“语义”部分)。
通常,我可以用结构名称命名文件,并再次使用该名称来选择 header 保护定义。
如果您需要使用指向结构的指针来声明函数,则不需要完整的结构定义。一个简单的前向声明,例如:
struct a ;
就足够了,它减少了耦合。
这是另一种方式,更简单一些,但模块化程度较低:一些只需要您的结构即可工作的代码仍然必须包含所有类型。
在 C++ 中,这可能会导致有趣的并发症,但这是题外话(没有 C++ 标记),所以我不会详细说明。
也许我没明白重点,但 Greg Hewgill 在他的帖子 How to declare a structure in a header that is to be used by multiple files in c? 中有一个很好的答案.
原因是 C 结构管理可能很痛苦:您必须在使用它的任何地方声明 struct 关键字:
struct MyStruct ; /* Forward declaration */
struct MyStruct
{
/* etc. */
} ;
void doSomething(struct MyStruct * p) /* parameter */
{
struct MyStruct a ; /* variable */
/* etc */
}
虽然 typedef 将使您能够在不使用 struct 关键字的情况下编写它。
struct MyStructTag ; /* Forward declaration */
typedef struct MyStructTag
{
/* etc. */
} MyStruct ;
void doSomething(MyStruct * p) /* parameter */
{
MyStruct a ; /* variable */
/* etc */
}
重要您仍然保留结构的名称是很重要的。写作:
typedef struct
{
/* etc. */
} MyStruct ;
只会创建一个带有 typedef 名称的匿名结构,您将无法转发声明它。所以请保持以下格式:
typedef struct MyStructTag
{
/* etc. */
} MyStruct ;
因此,您可以在任何您想避免添加 struct 关键字的地方使用 MyStruct,并且在 typedef 不起作用时仍然使用 MyStructTag(即前向声明)
更正了关于 C99 结构声明的错误假设,正如 Jonathan Leffler 正确指出的那样.
Craig Barnes在他的评论中提醒我们,您不需要为结构“标签”名称及其“typedef”名称保留单独的名称,就像我在上面为清楚起见所做的那样。
事实上,上面的代码完全可以写成:
typedef struct MyStruct
{
/* etc. */
} MyStruct ;
IIRC,这实际上是 C++ 在幕后对其更简单的结构声明所做的,以保持它与 C 的兼容性:
// C++ explicit declaration by the user
struct MyStruct
{
/* etc. */
} ;
// C++ standard then implicitly adds the following line
typedef MyStruct MyStruct;
回到 C,我已经看到了两种用法(单独的名称和相同的名称),而且我知道没有一个有缺点,所以如果您不使用 C separate "namespaces" for structs and other symbols,使用相同的名称会使阅读更简单.
关于c - 如何在头文件中声明一个结构,该结构将被 c 中的多个文件使用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/228684/
我有一位客户希望她的网站适合 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
我是一名优秀的程序员,十分优秀!