数组的外部声明?

问题描述 投票:0回答:7

我在一个文件中定义了一个数组,而在另一个文件中我必须使用它,例如-

/* a.c - defines an array */

int a[] = {1,2,3,4,5,6,7,8,9}; 


/* b.c - declare and use it. */

#define COUNT ((sizeof a)/(sizeof int))
extern int a[];  //size of array

.
.
.

int i;
for(i=0; i<COUNT; i++)
  printf("%d", a[i]);
.
.
.

现在,当我尝试编译它时,它给了我错误,说 sizeof 不能用于不完整的类型。

有人可以告诉我如何在 C/C++ 中处理这种情况吗?我不想在 a.c 中数组下标

提前致谢

c
7个回答
29
投票

您可以将以下内容放入

a.c
中,然后从
b.c
中将其外部。

交流电中:

int a[] = {1, 2, 3};
const int lengthofa = sizeof( a ) / sizeof( a[0] );

然后在公元前:

extern int a[];
// the extern (thanks Tim Post) declaration means the actual storage is in another 
// module and fixed up at link time.  The const (thanks Jens Gustedt) prevents it
// from being modified at runtime (and thus rendering it incorrect).
extern const int lengthofa;

void somefunc() {
  int i;
  for ( i = 0; i < lengthofa; i++ )
    printf( "%d\n", a[i] );
}

16
投票

如果您希望数组大小可以作为编译时常量访问,那么您别无选择,只能在数组的

extern
声明中显式指定数组大小

extern int a[9];

在这种情况下,您有责任确保数组大小在

extern
声明和定义之间保持一致。您可以为此使用清单常量,但您仍然有责任确保
{}
和声明的大小之间的初始值设定项数量相同。

如果您不介意将数组大小作为编译时常量,那么您可以按照 Mark Wilkins 在他的答案中建议的操作。


2
投票

我会在通常用于定义此类内容的外部标头中

#define ARRAY_MAX (or whatever)
,然后将该标头包含在需要它的两个文件中。当您倾向于将大部分(如果不是全部)
define
保留在一个或两个中央标题中时,此方法非常有效。


1
投票

编译器在编译 b.c 时没有足够的信息来确定数组的大小。 该信息仅存在于您的初始值设定项列表范围内的 a.c 中。

您必须以某种方式传达尺寸。 一种方法是定义一个具有大小的 int const 和 extern。 另一种可能性是使用哨兵(有效数据范围之外的值)来指示数组的末尾。


0
投票

在我看来,如果您没有定义并在一个文件中使用 sizeof a 进行定义,则它无法编译。

文件被编译并存储为 *.obj / *.a 文件。由于 extern 声明,您可以使用其他文件中的数组,该声明将在编译后在链接过程中进行检查。

您想要声明定义(预处理器必须在此提供帮助。它在编译器之前运行)。

所以在编译之前你不会从另一个文件中获取数组...


0
投票

仅将变量放入头文件中,如下所示:

#ifndef A_DEF
#define A_DEF
int a[] = {1,2,3,4,5,6,7,8,9}; 
#endif

条件编译不允许变量重新定义。因此您可以从任何地方访问它。


0
投票

马克的回答很棒。但是,如果由于某种原因需要编译时常量

COUNT
,则可以定义
COUNT
,然后使用
static_assert

检查它
/* a.h */    
#define COUNT 9
extern int a[];

/* a.c */    
int a[] = {1,2,3,4,5,6,7,8,9}; 
static_assert(sizeof a / sizeof a[0] == COUNT, "remember to update COUNT");

/* b.c */    
int i;
for(i=0; i<COUNT; i++)
  printf("%d", a[i]);
© www.soinside.com 2019 - 2024. All rights reserved.