2

今天早些时候,我浏览了各种头文件,只是为了将它们与我正在制作的头文件进行比较,并注意到它们似乎声明它们的函数有点不同。

例如这里是strlenfrom的声明string.h

extern size_t __cdecl strlen(const char *);

在做了一些研究后,我发现这extern是为了在功能块之外声明一个变量。在头文件中声明我的函数也是最佳做法extern吗?

我看到他们使用size_twhich is unsigned long longhere 而不是int,我假设这是因为它更有效有几个原因(例如字符串的长度永远不会是负数),但这是他们size_t在这里使用的原因吗?还是我完全错过了重点?

然后最后我看到__cdecl我找不到太多信息。究竟是__cdecl什么?我也应该使用它吗?

最后,我注意到在这个声明中没有变量名用于传递给strlen. 我猜这是因为这不是一个函数原型,只是一个声明,而原型在别处。为什么声明中没有变量名strlen(const char *str)

我的最后一个问题是,如果这只是一个声明,strlen 的函数原型会是什么样子?我的猜测是这样的:

size_t strlen(const char *str)

我只是问,因为我想学习和改进我的代码(假设我在 C 文件中制作函数原型/声明,然后只是在头文件中声明函数,以便其他 C 文件可以使用它们)。

4

4 回答 4

6
  1. size_t是更合适的返回值strlen而不是int
  2. __cdecl是函数的调用约定。这表示谁为参数、返回值等设置堆栈以及谁清除它。更多参考:调用约定
  3. 在声明函数时,您实际上并不需要参数名称。只需参数类型就足够了。

更新extern

  • extern告诉编译器该语句只是一个声明而不是定义。因此,对于函数原型,extern不会添加任何价值,因为它已经只是一个定义。参考:C 外部

希望这可以帮助。

于 2012-08-27T17:40:26.763 回答
1

声明函数时关键字是多余的extern,因为很明显它必须在其他地方定义。对于没有extern.

于 2012-08-27T17:54:46.750 回答
1

size_t 定义strlen返回的内容。在您的情况下,它似乎也像一个 64 位系统,理论上一个字符串可以大于可以存储在int.

__cdecl是编译标准库时使用的调用对流。如果您应该为程序选择不同的调用约定(使用一些编译器选项),预编译的库函数仍将使用正确的约定。

我经常在函数声明中命名所有参数,因为它有助于记录它们是什么。只是int f(int, int, int, int)对我帮助不大,但对编译器来说已经足够了。因此名称是可选的。

于 2012-08-27T17:56:59.000 回答
1

正如@Rohan 所说,关于 size_t ,它是一种用于保存大小的类型。例如,它不能是负面的。这主要是因为一些安全问题。(例如getpeername,FreeBSD 中使用 int 而不是 size_t 导致的安全漏洞)
关于 cdecl 这可能会对您有所帮助(它来自PC 汇编手册):

Borland 和 Microsoft 使用通用语法来声明调用约定。他们将 cdecl 和 stdcall 关键字添加到 C 中。这些关键字充当函数修饰符,并立即出现在原型中的函数名称之前。例如,对于 Borland 和 Microsoft,函数 f 将定义如下:

无效 __cdecl f ( int );

每个调用约定都有优点和缺点。cdecl 约定的主要优点是简单且非常灵活。它可以用于任何类型的 C 函数和 C 编译器。使用其他约定可能会限制子例程的可移植性。它的主要缺点是它可能比其他一些慢并且使用更多内存(因为函数的每次调用都需要代码来删除堆栈上的参数)。

stdcall 约定的优点是它使用的内存比 cdecl 少。在 CALL 指令之后不需要进行堆栈清理。它的主要缺点是它不能与具有可变数量参数的函数一起使用。[喜欢printf]

于 2012-08-27T17:58:12.277 回答