侯捷C++八部曲笔记(六、C++程序的生前死后)

侯捷C++八部曲笔记(六、C++程序的生前死后)

生前:CRT startup code

看完课程,能够回答一下问题:

  1. C++进入点是main()嘛?
  2. 什么代码比main更早执行?
  3. 什么代码在main结束后执行?
  4. 为什么上述代码可以如此行为?
  5. Heap的结构如何?
  6. I/O的结构如何?

startup code

有一个启动代码:就是指定/ENTRY:function (Entry Point Symbol),他是一个function,函数的调用形式,返回值都有规定。需要完成的事:初始化CRT library,初始化静态对象。

main应该由启动代码来调用。

在使用gcc编译,-e参数用于指定启动函数。

还记得内存管理中的内存分配的具体步骤嘛?

在这里插入图片描述

在内存管理部分,我们只关注前面两个函数,他们完成指定的内存分配策略。

  1. _heap_init(): 内存heap初始化。内存管理那一节我们已经讲过,是做16个header初始化工作。
  2. _ioinit():IO初始化,malloc分配256k大小(过程很复杂,见内存管理那个部分),用于IO初始化。比如初始化stdin(0)、stdout(1)、stderr(2)等东西(最多分配2048个文件描述符),不太感兴趣哦。

environ是pointer to pointer table,table中的每个entry都是代表环境变量的pointer to string,如下图:
在这里插入图片描述
也就是要得到main函数的参数,包括:__argc,__argv,_environ。其中的_environ中的内容就是startup code想要得到的,一般由操作系统持有,程序需要把这个信息从操作系统copy过来,所以分配内存来存放这些信息。环境变量具体的内容如下:
在这里插入图片描述

  1. GetCommandLineA(): 处理一个字符串。
  2. __crtGetEnvironmentStringsA(): 处理一堆字符串
  3. _setargv(): 命令行的参数
  4. _setenvp(): 引发11次内存分配,也就是上面说的,将操作系统持有的环境变量复制到程序中来,需要内存来存放。
  5. _cinit():

上面的3-7步骤的详细内存分配计算图如下,每个分配的内存大小都需要膨胀,加debug信息32bytes,加cookie 8bytes,都是在由ioinit申请的内存中分配的:

在这里插入图片描述

HeapCreate

前面说到,小于1k的内存交给SBH来分配,其余的内存交由操作系统来分配,也就是HeapCreate来处理。

有128条链表用于管理内存分配,设计思想和SBH内存分配是一样的,只不过能分配的内存更大,对应的cookie只有上cookie,上cookie中存放当前块大小,前一个区块大小,单位为单元(一个单元8byte),而不是字节。

总结

main执行之前

  1. 设置栈指针:为栈分配相关的位置,用来放一些局部变量和其他数据
  2. 初始化static静态和global全局变量,即data段的内容:把全局和静态变量初始化,放在相应的位置
  3. 将未初始化部分的全局变量赋初值:数值型short,int,long等为0,bool为FALSE,指针为NULL,等等,即.bss段的内容:将未设置初值的全局变量赋初值
  4. 全局对象初始化,在main之前调用构造函数
  5. 将main函数的参数,argc,argv等传递给main函数,然后才真正运行main函数:argc为整数,argv为指针的指针

main执行之后

执行全局的析构函数,可以用_onexit 注册一个函数,它会在main 之后执行; 。

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值