为什么在main之后有函数定义?

如何解决为什么在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 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 <select id="xxx"> SELECT di.id, di.name, di.work_type, di.updated... <where> <if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 <property name="dynamic.classpath" value="tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams['font.sans-serif'] = ['SimHei'] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -> systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping("/hires") public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate<String
使用vite构建项目报错 C:\Users\ychen\work>npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-