如何解决为什么在main之后有函数定义?
| 我认为由最权威的程序员之一(Richard Stallman)编写的最著名的OS(linux)之一中使用最多的系统功能(ls
)可能是编写得很好的代码的一个示例。
因此,由于它是开源的,所以我决定看一下代码(例如参见此处)。我在found1ѭ之后找到了几个定义的函数,因此在调用它们之后,这些函数我并不常见。
有经验的C程序员会对此发表评论吗?
解决方法
斯托曼在这里所做的工作绝对没有错。
C语言允许向前声明将在以后定义的函数。
这具有许多优点,不应被视为不良行为,而应被视为非常良好的行为。
优点(并不详尽):
-让程序员快速了解C代码公开的API,而不必查看所有代码
-允许使用头文件,您可以在其中声明一个函数,该函数将在以后的编译过程中定义。这样您就不必每次使用时都定义函数。
对于这种“ 0”实现,他只是简单地声明了将在“ 1”中使用的功能,但是如果仔细看,主要功能是第一个出现的功能。
这很可能是出于可读性的考虑,因此您不必一直向下滚动即可到达程序的入口点。
请注意,这里的词汇很重要:
-函数声明的意思是:只是告诉编译器,将在代码中的某个位置定义具有相同名称的函数。
-函数定义:实际的函数实现
int my_function( char *text); // function declaration,no implementation
int main( int argc,char **argv)
{
return my_function(argv[0]); // use of the declared function
}
// actual function definition / implementation
int my_function( char *text )
{
printf(\"%s\\n\",text);
}
编辑:仔细查看代码后,您可以看到Stallman没有向前声明其所有功能。他还有一种很奇怪的定义功能的方式。我将其归因于代码的过时,该时间可追溯到1985年,当时C编译器的定义不如今天。
在声明或定义之前,它必须允许这种功能使用。
最后但并非最不重要的一点是,可以在此处找到ls
源代码的最新版本:http://coreutils.sourcearchive.com/documentation/7.4/ls_8c-source.html,
比\ '85(最新版本)版本具有更多的C99兼容编码。
, 在C语言中,通常在头文件中定义函数原型,然后包括头文件,因此可以在调用函数后安全地定义它们。
在您提供的示例文件中,程序足够小,可以将原型简单地放在过程声明之前的文件顶部,但适用相同的原理。
编辑:而且,该文件是K&R C之前的版本,这有点酷,但与现代C有一些显着差异,您不必模仿它。
, 该代码从概念上看起来像是一个不错的规范
首先定义主要入口点使用的接口
然后定义主入口点
然后,从概念上完全指定了理想情况下程序的行为。
最后,实现接口(main
之后的定义)。
对于现代C语言,许多程序员都认为他省略了要放在ѭ6之前的函数的声明(以满足第一个项目符号-定义接口),这被认为是不好的风格。
但是,代码使用旧式函数定义,其声明未定义调用方的参数类型。据推测,将代码更新为现代C语言会破坏许多仍使用ANSI之前的编译器的非常老的系统。
, 对我来说,这是前原型时代的遗物,看看新代码,您常常会发现代码顺序颠倒了。对我来说,主要的缺点是必须先声明该函数,然后将该函数本身向下声明。如果您更改功能的签名,则还必须在2个位置进行更改。请注意,静态函数也是如此,这是这段代码中完全没有使用的功能...
当然,如果不这样做的话,它将可以编译,但是您早晚会被C的隐式声明所困扰。
认真地说,发现自己的代码比旧的K&R风格C更新,C语言从那时起得到了很大发展,您可能会在此过程中养成一些不良习惯。
,调用该函数之前,您需要在作用域中拥有一个原型。函数定义用作原型
/* this is a definition and a prototype */
int fx1(void) {
return 42;
}
/* this is only a prototype */
int fx2(int,const char*);
int main(void) {
fx1(); /* ok,prototype in scope */
fx2(42,\"foobar\"); /* ok,prototype in scope */
}
/* fx2 definition.
** it is undefined behaviour if the prototype here
** does not match the previous prototype */
int fx2(int k,const char *t) {
return strlen(t) - k;
}
原型通常在头文件中声明,头文件包含在实现文件的顶部,而不是任何代码。
#include \"prototypes.h\"
/* define functions in any order: prototypes are all in scope */
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。