与 extern 关键字使用相关的疑问

发布于 2024-08-23 08:36:24 字数 290 浏览 7 评论 0原文

AFAIK,应该使用 extern 关键字进行声明,并且任何值都不能与使用 extern 关键字声明的变量关联。但是假设我写了一条语句,例如

extern int i = 10;

编译器是否应该标记相同的错误?我见过一些编译器容忍并忽略这一点?为什么会这样呢? “C”标准对此有何规定?

编辑:@All,感谢您的回答。但我仍然有疑问。假设我在另一个文件(例如 ac)中定义了这个变量,而没有外部链接,并且我在 bc 中添加了这个语句,那么编译器不标记错误是否可以?它会被重新定义吗?

AFAIK, extern keyword should be used for declaration and no value can be associated with the variable being declared with extern keyword. But supposing I write a statement like

extern int i = 10;

Should the compiler flag an error for the same? I have seen some compilers being tolerant and ignoring this? Why is this so? What does the 'C' standard says about this?

EDIT: @All, Thanks for the answers. I have a doubt still though. Suppose I have the definition for this variable without the extern linkage in another file say a.c and I add this statement in b.c. Still is it Ok for the compiler not to flag an error? Does it come under redefintion?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(4

叹倦 2024-08-30 08:36:24

这是有效的语法,C99 标准中甚至有一个基本相同的示例。 (参见第 6.9.2-4 节。)

这些示例确实不是规范性的,但我相信它的目的是成为合法的语法。编译器经常会输出警告,因为它并没有真正完成任何事情。

4 示例 1

int i1 = 1;             // definition, external linkage
static int i2 = 2;      // definition, internal linkage
extern int i3 = 3;      // definition, external linkage
int i4;                 // tentative definition, external linkage
static int i5;          // tentative definition, internal linkage
int i1;                 // valid tentative definition, refers to previous
int i2;                 // 6.2.2 renders undefined, linkage disagreement
int i3;                 // valid tentative definition, refers to previous
int i4;                 // valid tentative definition, refers to previous
int i5;                 // 6.2.2 renders undefined, linkage disagreement
extern int i1;          // refers to previous, whose linkage is external
extern int i2;          // refers to previous, whose linkage is internal
extern int i3;          // refers to previous, whose linkage is external
extern int i4;          // refers to previous, whose linkage is external
extern int i5;          // refers to previous, whose linkage is internal

That's valid syntax, there is even an essentially identical example in the C99 standard. (See §6.9.2-4.)

It's true that the examples are not normative but I believe it was intended to be legal syntax. The compiler will often output a warning, because it doesn't really accomplish anything.

4 EXAMPLE 1

int i1 = 1;             // definition, external linkage
static int i2 = 2;      // definition, internal linkage
extern int i3 = 3;      // definition, external linkage
int i4;                 // tentative definition, external linkage
static int i5;          // tentative definition, internal linkage
int i1;                 // valid tentative definition, refers to previous
int i2;                 // 6.2.2 renders undefined, linkage disagreement
int i3;                 // valid tentative definition, refers to previous
int i4;                 // valid tentative definition, refers to previous
int i5;                 // 6.2.2 renders undefined, linkage disagreement
extern int i1;          // refers to previous, whose linkage is external
extern int i2;          // refers to previous, whose linkage is internal
extern int i3;          // refers to previous, whose linkage is external
extern int i4;          // refers to previous, whose linkage is external
extern int i5;          // refers to previous, whose linkage is internal
潦草背影 2024-08-30 08:36:24

以下代码;

extern int i ;

声明一个变量 i,但不实例化它。如果它没有在同一编译单元中定义,则链接器将尝试从构成最终可执行文件的目标文件和库中解析它。

但是,您的示例:

extern int i = 10 ;

初始化对象,因此也必须实例化它。在这种情况下,extern 关键字是多余的,因为对象是在同一编译单元(实际上是同一语句)中初始化的。它相当于:

extern int i ;  // redundant
int i = 10 ;

虽然在最后一个示例中 extern 关键字是多余的,但它与在头文件中声明全局变量并实例化时所拥有的完全相同> 在还包含该标头的源文件中(因为它应该允许编译器执行类型检查)。

您可以按如下方式测试:

extern int i ;
int main()
{
    i = 10 ;
}

对于未解析的变量 i,上述内容将导致链接器错误。而:

extern int i = 10 ;
int main()
{
    i = 10 ;
}

将毫无问题地链接。

The following code ;

extern int i ;

declares a variable i, but does not instantiate it. If it is not also defined in the same compilation unit, the linker will attempt to resolve it from the object files and libraries that comprise the final executable.

However your example:

extern int i = 10 ;

initialises the object, and therefore must also instantiate it. In this case the extern keyword is redundant because the object is initialised in the same compilation unit (in fact the same statment). It is equivalent to:

extern int i ;  // redundant
int i = 10 ;

Although in this last example the extern keyword is redundant, it is exactly equivalent to what you have when a global variable is declared in a header file, and instantiated in a source file that also includes that header (as it should, to allow the compiler to perform type checking).

You can test this as follows:

extern int i ;
int main()
{
    i = 10 ;
}

The above will cause a linker error for unresolved variable i. Whereas:

extern int i = 10 ;
int main()
{
    i = 10 ;
}

will link without problem.

漆黑的白昼 2024-08-30 08:36:24

extern 关键字指示给定变量分配在不同的模块中。它与对该变量的访问无关。分配给外部变量是完全合法的。

The extern keyword indicates that the given variable is allocated in a different module. It has nothing to do with access to that variable. It's perfectly legal to assign to assign to an extern variable.

始终不够 2024-08-30 08:36:24

extern 关键字的目的是为实体提供外部链接。无论它用在声明还是定义中都没有区别。您发布的代码绝对没有错误。

如果您更愿意从“导出与导入”的角度来考虑,那么应用于非定义声明的 extern 关键字意味着我们导入< /em> 在其他翻译单元中定义的实体。当 extern 关键字应用于定义时,意味着我们导出该实体以供其他翻译单元使用。 (尽管值得注意的是“导出与导入”并不完全是思考 C 链接概念的标准方式。)

您不会经常看到定义中使用关键字的原因是因为在 C 文件范围中默认情况下,定义具有外部链接。因此,编写

extern int i = 10;

是有效的,但是多余的,因为它相当于普通的

int i = 10;

然而,在实际代码中,您有时可能会看到人们在函数声明和定义中使用此关键字,即使它在那里也是多余的

extern void foo(int i); /* `extern` is superfluous */
...
extern void foo(int i) /* `extern` is superfluous */
{
  /* whatever */
}

The purpose of extern keyword is to give the entity external linkage. Whether it is used in a declaration in a or definition makes no difference. There's absolutely no error in the code you posted.

If you prefer to think about it in terms of "export vs. import", then extern keyword applied to a non-defining declaration means that we are importing an entity defined in some other translation unit. When extern keyword applied to a definition, it means that we are exporting this entity to be used by other translation units. (Although it is worth noting that "export vs. import" is not exactly a standard way of thinking about the concept of C linkage.)

The reason you won't see the keyword used in definitions very often is because in C file-scope definitions have external linkage by default. So writing

extern int i = 10;

is valid, but redundant, since it is equivalent to plain

int i = 10;

Yet, from time to time in the actual code you might see people using this keyword with function declarations and definitions, even though it is superfluous there as well

extern void foo(int i); /* `extern` is superfluous */
...
extern void foo(int i) /* `extern` is superfluous */
{
  /* whatever */
}
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文