上一篇文章,大概了解了链接器的工作内容就是:符号解析和重定位。点击上一篇文章查看:点击查看

本片文章其实还是围绕链接器来学习。只不过不是很明显,当你学到下一篇文章时,就明白了。

本篇文章来弄明白一个问题:在C/C++程序被加载到内存中准备运行时,main函数是第一个被执行的函数么?答案肯定不是!如果是,就没必要去写这个文章了!!!

1、_start()函数

先说结果:_start() 是第一个被执行的函数,而不是main()函数。

我们的程序中并没有写_start() ,它是通过链接器链接到可执行文件中的(下一篇文章可以学习到这是如何办到的)。

在默认情况下(gcc)

  • 程序加载后,_start()是第一个被执行的函数
  • _start()函数,准备好参数后,立即调用 __libc_start_main() 函数
  • __libc_start_main() 初始化运行环境后,调用main()函数执行

注意:_start()函数的入口地址就是代码段(.text)的起始地址

2、__libc_start_main()函数的作用

__lib_start_main()函数的作用

  • 调用__libc_csu_init()函数(完成必要的初始化操作)
  • 启动程序的第一个线程(主线程),main()函数为主线程入口
  • 注册__libc_csu_fini()函数(程序运行终止时被调用)

3、程序的启动过程

下面给一个简图,来说明一个C/C++程序的执行过程:

4、自定义程序的入口函数

  • gcc 提供-e选项,用于在链接时指定入口函数
  • 自定义入口函数时,必须使用选项 -nostartfiles 进行链接

比如以下程序:

program.c

#include <stdio.h>
#include <stdlib.h>

int program()
{
    printf("D.T.Software\n");
	
    exit(0);
}

很明显它没有main函数。但是我们通过以下编译命令进行编译:

  • gcc -e program -nostartfiles program.c -o program

生成可执行文件 program

运行该可执行文件: ./program

可以得到运行结果:

显而易见,我们改变了程序的入口函数。虽然没有写main函数,但是依然可以运行该程序!!!

5、总结

本文主要是理解程序的入口函数。最好自己私底下做实验,看一下可执行文件的反汇编代码,就知道整个程序的执行流程了。

本文参考狄泰软件学院相关课程
想学习的可以加狄泰软件学院群,
群聊号码:199546072

学习探讨加个人(可以免费帮忙下载CSDN资源):
qq:1126137994
微信:liu1126137994