问题引入
引入问题表格
何时运行 | 构造函数 | 析构函数 |
---|---|---|
EXE中的全局对象 | ? | ? |
DLL中的全局对象 | ? | ? |
回答
何时运行 | 构造函数 | 析构函数 |
---|---|---|
EXE中的全局对象 | C运行时Startup代码 | C运行库DLL清理 |
DLL中的全局对象 | C运行时优先于DllMain的DLL_PROCESS_ATTACH函数 | C运行时后于DLL_PROCESS_DETACH返回 |
EXE中的全局对象构造
在可执行文件中,全局对象的构造总所皆知。因为C运行时的启动代码会先链接,并在调用正式入口点函数(main或者wWinMain)之前做很多准备。
期中之一的准备工作就是调用全局对象的构造函数。由于C运行时会负责这些准备工作,所以全局对象的构造就是在这个时候完成的。
DLL的全局对象构造与析构
DLL的全局对象也和EXE的全局对象类似。如同main函数一样,DllMain函数的入口点并不是实际的DLL入口函数,真正的入口点函数也是由C运行时提供的,全局对象的构建也是在这个时候完成的,然后才会调用DllMain函数载入。
这些准备工作,构建全局对象,是得DLL_PROCESS_ATTACH函数中构建的。
BOOL CALLLBACK RealDllMain(HINSTANCE hinst, DWORD dwReason, void *pvReserved){
……
case DLL_PROCESS_ATTACH:
Initialize_C_Runtime_Library();
Construct_DLL_Gloable_Objects();
DllMain(hinst, dwReson, pvResered);
……
case DLL_PROCESS_DETACH:
DllMain(hinst, dwReason, pcReserved);
Destruct_DLL_Global_Objects();
Uninitialize_C_Runtime_Library();
break;
……
}
当然,实际代码并不仅限于此,这里只是大致抽象了一下。
EXE中的全局对象析构
这里需要首先了解下C运行库
例如: MSVCR80.dll
同时,还要了解一个函数 atexit
int atexit(
void (__cdecl *func)(void)
);
这里的func是指要调用的函数,成功返回0,失败返回非0
上述atexit函数传递函数的地址Func,会在程序正常退出的时候被调用。在启动时,连续调用atexit会注册一个函数寄存列表,并以先进先出的方式,在退出时候执行。
atexit和_one