87682715 2019-06-29
下面的 main 函数定义正确吗?
A.
main()
{
}B.
void main()
{
}c.
int main()
{
}D.
int main()
{
return 0;
}A B C D 编译无警告,无错误,运行正常
D 为最标准
思考:为什么 C 编译器支持那么多不同的 main 函数元原型呢?
C 语言诞生初期就得到的广泛的应用,,当时的程序比较简单,主要应用于科学运算或嵌入式设备中。在当时的环境下,只需要知道入口函数在哪里便可以,在非操作系统中运行,返回值是无意义的。
随着商业编译器的竞争与发展,编译器开始支持更多的 C 语言特性,并对编译器的功能进行扩展。
A.C
#include <stdio.h>
int main()
{
printf("I'm A!\n");
return 0;
}B.c
#include <stdio.h>
int main()
{
printf("I'm B!\n");
return 99;
}命令行:【GCC】 ./a.out && ./b.out 输出: I'm A! I'm B! 命令行:【GCC】 ./b.out && ./a.out 输出: I'm B!
在操作系统中的作用
操作系统中存在批处理语言。在 linux 中为 shell 脚本,其联合各种应用程序协作完成任务,通过应用程序返回值监控程序的运行状态。
int mainint main(int argc)int main(int argc, char* argv)int main(int argc, char* argv[], char* env)
argc - 命令行参数个数
argv - 命令行参数数组 (命令行参数以字符串形式”存储“于指针数组中)
env - 环境变量数组 (环境变量以字符串形式“存储”于指针数组中)
./a.out a.c b.c c.c
argc ---> 4 argv[0] ---> a.out argv[1] ---> a.c argv[2] ---> b.c argv[3] ---> c.c
#include <stdio.h>
int main(int argc, char* argv[], char* env[])
{
int i = 0;
printf("=============Begin argv=============\n");
for(i=0; i<argc; i++)
{
printf("%s\n", argv[i]);
}
printf("=============End argv=============\n");
printf("\n");
printf("\n");
printf("\n");
printf("=============Begin env=============\n");
for(i=0; env[i]!=NULL; i++)
{
printf("%s\n", env[i]);
}
printf("=============End env=============\n");
}命令行: ./a.out a.c b.c c.c 输出: =============Begin argv============= ./a.out a.c b.c c.c =============End argv============= =============Begin env============= ORBIT_SOCKETDIR=/tmp/orbit-delphi SSH_AGENT_PID=1695 TERM=xterm SHELL=/bin/bash XDG_SESSION_COOKIE=6c560f89cd4609726ff940b800000007-1544093738.93069-2082860285 WINDOWID=71303204 ...... ...... =============End env=============
实例,gcc 命令行不同参数,实现不同功能
不一定。
在 gcc 中使用属性关键字 attribute, 可以指定 main 之前与 main 之后分别执行一个函数。
但有的编译器没有支持相关功能关键字扩展,那么 main 函数是第一个运行的函数,例BCC。
#include <stdio.h>
#ifndef __GNC__
#define __GNU__
#endif
__attribute__((constructor))
void before_main()
{
printf("%s\n", __FUNCTION__);
}
__attribute__((destructor))
void after_main()
{
printf("%s\n", __FUNCTION__);
}
int main()
{
printf("%s\n", __FUNCTION__);
return 0;
}输出: before_main main after_main
补充:不同编译器有不同的功能扩展,为了程序的可以执性,如果不是及其特殊情况,不编写依赖编译器的代码
以上内容参考狄泰软件学院系列课程,请大家保护原创!