一道海康的面试题,C语言的main函数有几个参数(多选)
A:argc
B:envp
C:main
D:argv
正确答案是:ABD
一般情况下,我们会认为是只有两个参数,因为从我们从第一次接触,基本看到的就一直是int argc,char *argv[]这样的形参,第一个参数argc是整型变量,第二个参数argv是指向字符串的指针数组,当然main函数也可以不写参数,这种情况比较少。
那么从实际给出的答案中,我们知道实际情况是还有一个指向环境变量的指针数组char *envp[]
我们先来看下 int argc 这个我们知道,是统计命令行参数的个数:
#include <stdio.h>
int main(int argc, char *argv[])
{
printf("参数个数:%d\n", argc);
for (int i = 0; i < argc; i++)
{
printf("%s\n", argv[i]);
}
return 0;
}
Windows环境:
>gcc test.c -o test
>test Hello China Great
参数个数:4
test
Hello
China
Great
Linux环境:
$ gcc test.c -o test
$ ./test Hello China Great
参数个数:4
./test
Hello
China
Great
可以看到上述的程序名test也当作了一个参数,所以我们获取参数,最好下面这样写:
#include <stdio.h>
int main(int argc, char *argv[])
{
while (argc-- > 1)
{
printf("%s\n", *++argv);
}
return 0;
}
前面两个参数大家熟悉了,重点来看下环境变量envp这个指针数组,正确的写法如下:
#include <stdio.h>
int main(int argc, char *argv[], char *envp[])
{
int i = 0;
while (envp[i])
{
printf("%s\n", envp[i++]);
}
return 0;
}
重新编译之后,Win和Linux中的截图如下:
可以看到显示了本地的环境变量,每个字符指针指向一条环境变量。
我们来了解下,这个main函数是怎么被调用的,首先会在加载时,启动一个_start函数,这个函数调用_libc_start_main函数,再去调用main函数的,也就是说,真正的入口函数应该是main被调用前的两个函数。
_start函数的作用主要就是调用__stack_init来初始化堆栈,调用__data_init来初始化全局变量,以及main函数和__exit_program退出程序,将控制权交给操作系统。
那么这个_libc_start_main也是程序入口,怎么理解呢?这个_start函数一般是在汇编中定义,操作系统加载程序的时候,控制权就交给_start函数,而_libc_start_main是C标准库中定义的(glibc),作用也是做一些初始化工作和调用main函数,这个函数是有编译器自动调用,跟程序员无关。
所以这里,我们来看下_libc_start_main函数的源码,当然这个内部函数跟操作系统和编译器有很大关系,代码大概如下:
int _libc_start_main(int (*main)(int, char**, char**), void (*exit)(void), void (*abort)(void), char** environ, void (*h)(void (*)(void)), void (*h_rt)(void (*)(void)), void (*thread_detach)(void)) {
// 初始化C库
libc_init();
// 用户的main函数
int result = main(argc, argv, environ);
// 清理工作
libc_cleanup();
// 返回main函数结果
return result;
}
这里我们可以看到,main函数的参数是三个,类型分别是int, char**, char**,函数体重自定义的main函数三个形参名称为argc, argv, environ,其中environ就是指向环境变量的指针数组,这样就解释了,main函数本质上是三个参数。
我们也可以看到一道看似简单的面试题,也可以引申出操作系统对于如何加载C程序的一些扩展知识。