首先,看一下这个例子(我为了举例编写的,不是真正的程序):
whatever.h
#ifndef WHATEVER_H
#define WHATEVER_H
void fill(void);
#endif
main.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;
}
whatever.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;
to
extern char *names[];
那么一切都很好。
有人能解释一下为什么会发生这种情况吗?如果gcc无法将extern char **names;
与main.c
中的一个链接起来,它不应该给我一个错误吗?如果它可以链接它们,为什么names
最终在whatever.c
中变成了NULL
?
此外,extern char **names;
和extern char *names[];
有什么区别?
我正在使用Linux下的gcc版本4.5.1。
更新
为了进一步调查这个问题,我改变了main.c
中names
的定义:
char *names[10] = {"1", "2", "3", "4", "5", "6", "7", "8", "9", "10"};
(在
whatever.c
中保留extern char **names;
)并且使用gdb
,我可以看到names
有一个值。如果我将该值转换为char *
并打印它,它会给我"1"
。(请注意,不是*names
是"1"
,而是(char *)names
)基本上,这意味着gcc已经成功地将
whatever.c
中的extern char **names;
与main.c
中的names[0]
链接起来了!