- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
首先,看这个例子(我为了举例而编造的,它不是一个真正的程序):
随便.h
#ifndef WHATEVER_H
#define WHATEVER_H
void fill(void);
#endif
主.c
#include <stdio.h>
#include "whatever.h"
char *names[10] = {NULL};
int main()
{
int i;
fill();
for (i = 0; i < 10; ++i)
printf("%s\n", names[i]);
return 0;
}
随便.c
#include "whatever.h"
extern char **names;
void fill(void)
{
int i;
for (i = 0; i < 10; ++i)
names[i] = "some name";
}
当我制作这个程序时使用:
gcc -o test main.c whatever.c -Wall -g
我没有收到任何错误或警告。但是,当我运行该程序时,我在 fill
中看到, names
实际上是NULL
.如果在 whatever.c
我改变
extern char **names;
到
extern char *names[];
那么一切都很好。
谁能解释为什么会这样?如果 gcc 无法链接 extern char **names;
与 main.c
中的那个,它不应该给我一个错误吗?如果它可以链接它们,怎么会names
最终成为 NULL
在whatever.c
?
此外,extern char **names;
是如何实现的?不同于 extern char *names[];
?
我在 Linux 下使用 gcc 版本 4.5.1。
为了进一步研究这个问题,我更改了 names
的定义在main.c
到:
char *names[10] = {"1", "2", "3", "4", "5", "6", "7", "8", "9", "10"};
(将 extern char **names;
保留在 whatever.c
中)并使用 gdb
,我可以看到 names
有一个值。如果我将该值转换为 char *
打印出来,它给了我 "1"
. (请注意,这不是 *names
即 "1"
,而是 (char *)names
)
基本上,这意味着 gcc 以某种方式设法链接了 extern char **names;
。在whatever.c
与 names[0]
在main.c
!
最佳答案
每当您在不同的编译单元中对同一个变量使用不同的、不兼容的类型(就像您在此处所做的那样)时,就会出现未定义的行为。这意味着它可能无法正常工作,您可能不会收到任何关于它的错误或警告消息。
为什么会发生这种情况(以及为什么规范说这是未定义的)是由于大多数链接器的工作方式。大多数链接器对类型一无所知;他们只懂名字和内存。因此,就链接器而言,变量只是从某个地址开始的一 block 内存。在您的(原始)程序中,main.c
将 names
定义为指代内存块的开始,该内存块大到足以容纳 10 个指针(可能是 40 或 80 个字节,具体取决于无论这是 32 位还是 64 位系统),所有这些都是 NULL。另一方面,whaterver.c
假定 names
指的是一 block 大到足以容纳一个指针的内存块,并且该指针指向一个包含 10 个指针的数组。
关于c - `char *array[size]` 和 `extern char **array` 的错误链接?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10802959/
这段代码是否正确? extern "C" extern int x; // 1 extern extern "C" int y; // 2 extern "C" extern
根据 C++ Primer ,我们可以为定义为 extern 的变量提供初始化程序,但这样做会覆盖 extern。具有初始值设定项的 extern 是一个定义: extern double pi =
使用 Cuda 5.0、VS2010 这个项目在 VS2012 中编译和链接很好,但是 VS2012 不支持 Nsight 调试所以我也在 VS2010 中开发。所以我有一个 VS2010 项目文件,
这个问题已经有答案了: How do I use extern to share variables between source files? (19 个回答) Different compilat
我正在编写供 C 程序使用的 C++ 共享库。但是,我对 extern 和 extern "C" 有疑问。 考虑以下代码 我的头文件是这样的: #ifdef __cplusplus
我对整个 header 使用 extern "C" 说明符,还是为每个函数指定 extern 有区别吗? 据我所知,没有,因为只有函数和变量可以外部链接,所以当我在每个函数原型(prototype)和
这个问题在这里已经有了答案: What is the effect of extern "C" in C++? (17 个答案) 关闭 7 年前。 我见过 C/C++ 代码使用在函数签名中声明的 e
所以我使用 svn:externals 来检查一个外部仓库。外部仓库有自己的 svn-externals 设置。 现在,当更新我的项目的工作副本时,来自外部存储库的文件正在更新,但它的外部文件没有。该
是否可以忽略 svn:externals 属性中引用的标记的外部依赖性?这听起来像是一个很奇怪的问题,但让我解释一下...... 我收集了大量独立的“可插入”代码模块,每个模块都可以作为独立项目进行独
我见过 2 种创建全局变量的方法,有什么区别,什么时候使用它们? //.h extern NSString * const MyConstant; //.m NSString * const MyCo
我在 test 模块下通过 stripe api 在 stripe 中创建了一个帐户。并与该账户绑定(bind)一个银行账户。转到 Stripe dashboard -> connect -> acc
我在下面有一个代码。它是由 qemu 程序(一个 C 程序)使用 dlopen 加载的动态共享库的一部分。 extern "C" { extern uint64_t host_virt_offset;
C++ Primer 第 5 版第 60 页讨论了如何跨文件共享 const 变量 //file_1.cc extern const int bufSize = fcn(); //file_1.h e
这个 Unresolved external 问题有什么问题?我正在尝试将其实现到我的 MFC 应用程序的 InitInstance 中。但是我从调试器中收到此行错误。 LNK2019: unreso
在 C++ 中,extern(后面不跟语言链接字符串文字)似乎对命名空间范围 (Difference between declaration of function with extern and w
假设我有 3 个文件:file1.c、file2.c 和 globals.h。 file1.c 和 file2.c 都包含 globals.h。 file1.c 包含 file2.c 需要使用的结构。
我正在为具有 16 位安装程序的旧 CD-ROM 游戏编写新的安装程序,安装程序需要在硬盘上并且能够从原始光盘复制文件。如果所有游戏文件都打包在安装程序中,我已经设置了一个可以安装游戏的脚本,这适合个
在编译我的代码时,我收到此错误。 1>MSVCRTD.lib(crtexe.obj):错误 LNK2019:函数 ___tmainCRTStartup 中引用了未解析的外部符号 _main1>C:\U
我试图将 cimg 库包装在 c++/clr 中,当我尝试构建它时,我遇到了一堆链接错误。 Error 20 error LNK2028: unresolved token (0A0002AC)
我一直遇到这两个错误,但我似乎找不到有效的解决方案。 LNK1120: 1 unresolved externals Error 1 error LNK2019: unresolved externa
我是一名优秀的程序员,十分优秀!