为什么在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 */