C++ 初始化函数的实现
来源:互联网 发布:四年级上册英语软件 编辑:程序博客网 时间:2024/06/11 08:43
讲一点和实现细节相关的东西。在Visual C++中,所有在main之前执行的函数调用实际上都通过一个自动生成的函数来调用,比如下面这段代码:
int func(){ return 1;}int data = func();int main(){ return 0;}
实际上生成了三个函数:
?func@@YAHXZ,对应于 func
_main 对应于 main
_$E1 对应于 data=func() 这句赋值语句。它调用了 func,并且完成赋值这个操作。
窍门在于,VC将 _$E1这个函数的指针放到了段CRT$XCU中:
CRT$XCU SEGMENT
_$S2 DD FLAT:_$E1
; Function compile flags: /Odt /RTCsu /ZI
CRT$XCU ENDS
这个段的定义为:
CRT$XCU SEGMENT DWORD USE32 PUBLIC 'DATA'
CRT$XCU ENDS
参考 crt0dat.c 文件可以看到:
extern _CRTALLOC(".CRT$XIA") _PIFV __xi_a[];extern _CRTALLOC(".CRT$XIZ") _PIFV __xi_z[]; /* C initializers */extern _CRTALLOC(".CRT$XCA") _PVFV __xc_a[];extern _CRTALLOC(".CRT$XCZ") _PVFV __xc_z[]; /* C++ initializers */extern _CRTALLOC(".CRT$XPA") _PVFV __xp_a[];extern _CRTALLOC(".CRT$XPZ") _PVFV __xp_z[]; /* C pre-terminators */extern _CRTALLOC(".CRT$XTA") _PVFV __xt_a[];extern _CRTALLOC(".CRT$XTZ") _PVFV __xt_z[]; /* C terminators */
这里实际上有一个很巧妙的地方在于,VC应用了x86上段是连续并且可重叠的概念,因此CRT$XCU是位于CRT$XCA到CRT$XCZ之间,具体说,段的顺序是:
CRT$XCA SEGMENT DWORD USE32 PUBLIC 'DATA'CRT$XCA ENDSCRT$XCU SEGMENT DWORD USE32 PUBLIC 'DATA'CRT$XCU ENDSCRT$XCL SEGMENT DWORD USE32 PUBLIC 'DATA'CRT$XCL ENDSCRT$XCC SEGMENT DWORD USE32 PUBLIC 'DATA'CRT$XCC ENDSCRT$XCZ SEGMENT DWORD USE32 PUBLIC 'DATA'CRT$XCZ ENDS
由于CRT$XCA开始都是C++初始化函数,_PVFV实际上就是 void (*_PVFV)(),因此CRT的_initterm()函数就把这个段中的数据作为一个函数指针数组来访问,依次调用其中的函数,从而完成系统所有初始化操作。
最后,也是最关键的问题,就是实际上每个 CPP 文件编译好以后都有初始化函数,并且其指针位于 CRT$XC? 段中,随后连接程序 LINK 做了最后一个重要的任务,就是把所有具有相同名字的段合并成为一个单独的段(这也就是连接程序名字的由来之一),合并的做法就是简单地把每个段中的数据按顺序前后放到一个连续的空间就可以了。这样在最终运行的时候,程序看到的CRT$XC?段也就是一个连续的数组,而不是多个数组。
至于顺序问题,在这里就可以看到,是由连接程序最后做拼接时候确定的。连接程序拼接的顺序,基本上是它看到OBJ文件的顺序,也就是在连接程序命令行指定的顺序。因此,在程序中决不能依赖于这个顺序,因为在连接程序命令行中的文件顺序是不确定的。
以上是初始化程序的顺序,至于析构函数(或者在main函数之后的函数调用)则是通过用at_exit函数注册的顺序来确定,而注册往往是在初始化的时候进行,因此析构函数的调用顺序也是不确定的。
- C++ 初始化函数的实现
- 【C/C++】构造函数的初始化列表
- 顺序栈的C语言实现——初始化函数、入栈函数和出栈函数
- C语言:实现函数init()初始化数组、 实现empty()清空数组、 实现reverse()函数完成数组元素的逆置
- c语言:创建一个数组,分别实现函数初始化数组、 清空数组、数组元素的逆置
- C/C++—— 在构造函数中调用虚函数能实现多态吗(Vptr指针初始化的过程分析)
- c++: 构造函数(constructor)与各种眼花缭乱的初始化(initialization)
- C/C++的全局变量能否利用函数初始化?
- (C++)构造函数初始化
- C/C++ 通过初始化列表和构造函数内赋值初始化成员变量的区别
- C/C++ 通过初始化列表和构造函数内赋值初始化成员变量的区别
- C/C++ 通过初始化列表和构造函数内赋值初始化成员变量的区别
- 【深入理解C++】从初始化列表和构造函数谈C++的初始化机制
- C/C++ 通过初始化列表和构造函数内赋值初始化成员变量的区别
- C/C++ 通过初始化列表和构造函数内赋值初始化成员变量的区别
- 【深入理解C++】从初始化列表和构造函数谈C++的初始化机制
- 【深入理解C++】从初始化列表和构造函数谈C++的初始化机制
- 【深入理解C++】从初始化列表和构造函数谈C++的初始化机制
- iPhone开发之Category使用详解
- iphone-1
- perl 安装本地CPAN
- 漫谈虚函数
- DB2 sql中的内连接和外连接
- C++ 初始化函数的实现
- C# 使用委托跨线程通讯
- android中LayoutInflater的使用
- 学生管理系统流程分析图
- CSS中的zoom:1的作用
- Javascript之自定义事件
- vc Clistctrl中如何添加图片
- 2011-2012 年度总结
- 《Color Transfer Between Images》论文实现