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

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

侯捷C++八部曲笔记(六、C++程序的生前死后)
  • startup code
  • HeapCreate
  • 总结
    • main执行之前
    • main执行之后

生前: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 之后执行; 。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1352570.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-14
下一篇 2022-06-14

发表评论

登录后才能评论

评论列表(0条)

保存