3

我是外部概念的新手。今天在工作中,我遇到了大量在头文件中声明的外部函数;foo.h. 在一堆文件夹的某个地方,我发现了一个 foo.c 文件,其中包含所述函数的定义,但它没有#include foo.h。回到家后,我决定尝试使用外部存储类示例。在“C书”中做了一些阅读之后,这就是我想出的。

这是我没想到的。但确实如此。

主程序

#include <stdio.h> 

int data;

int main()
{

  data = 6;

  printf("%d\n", getData());

  data = 8;

  printf("%d\n", getData());
  return 0;
} 

外部人员.c

int getData()
{
  extern int data;
  return data;
}

重击

gcc main.c externs.c -o externs

我不认为这会起作用,因为该函数在 main 之前没有在技术上(或至少是冗长的)定义。这是否有效,因为默认存储类int getData()extern?如果是这样,为什么还要费心下面的例子(类似于我在工作中看到的)?

main2.c

#include <stdio.h> 
#include "externs.h"

int data;

int main()
{

  data = 6;

  printf("%d\n", getData());

  data = 8;

  printf("%d\n", getData());
  return 0;
} 

外部人员.h

#ifndef _EXTERNSH_
#define _EXTERNSH_

extern int getData();

#endif

外部人员.c

int getData()
{
  extern int data;
  return data;
}

重击

gcc main2.c externs.c -o externs
4

2 回答 2

5

函数是隐式的extern。您可以通过将它们声明为 来更改它static

你说得对,与它extern int getData();相比是多余的,int getData();但有些人认为它提高了代码的清晰度。

请注意,自 C99 以来,您的第一个示例格式不正确。它getData在没有声明该函数的情况下调用。在 C89 中,这执行了一个隐式int getData();声明,但从 C99 开始,它被删除了。使用编译器开关-std=c99 -pedantic对此进行测试。

于 2016-01-29T03:43:24.890 回答
4

默认情况下,GCC 4.x 或更早版本以 C89/C90 兼容模式编译,该模式不需要在使用前声明函数。未声明的函数被推断为返回类型为int. 默认情况下,GCC 5.x 或更高版本在 C11 兼容模式下编译,这确实需要在使用前声明函数 - 就像 C99 一样。

鉴于编译的代码没有抱怨,我们可以推断您使用的是 GCC 4.x。

即便如此,你也没有义务提供完整的原型,尽管你不这样做是愚蠢的。我经常使用以下选项编译代码:

gcc -std=c11 -O3 -g -Wall -Wextra -Werror -Wmissing-prototypes -Wstrict-prototypes \
    -Wold-style-definition -Wold-style-declaration …

我怀疑那里有一些冗余,但是在我开始运行代码之前,这些选项会发现很多问题。

于 2016-01-29T03:44:38.713 回答