青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

每天早晨叫醒你的不是鬧鐘,而是夢想

  C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
  62 Posts :: 0 Stories :: 5 Comments :: 0 Trackbacks

常用鏈接

留言簿(1)

我參與的團隊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

1、 在介紹靜態對象、全局對象與程序的運行機制之間的關系之前,我們首先看一下atexit函數。

atexit函數的聲明為:int atexit( void ( __cdecl *func )( void ) );

參數為函數指針,返回值為整型,0表示成功,其他表示失敗。當程序運行結束時,他調用atexit函數注冊的所有函數。如果多次調用atexit函數,那么系統將以LIFO(last-in-first-out)的方式調用所有的注冊函數。

舉例如下(代碼摘自MSDN):

 1#include <stdlib.h>
 2#include <stdio.h>
 3 
 4void fn1( void ), fn2( void ), fn3( void ), fn4( void );
 5 
 6void main( void )
 7{
 8  atexit( fn1 );
 9  atexit( fn2 );
10  atexit( fn3 );
11  atexit( fn4 );
12  printf( "This is executed first.\n" );
13}

14 
15void fn1()
16{
17  printf( "next.\n" );
18}

19 
20void fn2()
21{
22  printf( "executed " );
23}

24 
25void fn3()
26{
27  printf( "is " );
28}

29 
30void fn4()
31{
32  printf( "This " );
33}

34



編譯、運行程序后,程序的輸出為:
This is executed first.
This is executed next.
注冊函數的順序為:fn1、fn2、fn3、fn4,但是調用順序為fn4、fn3、fn2、fn1。

2、理解了atexit函數之后,我們就可以來看看局部靜態對象了。

1 class AAA{ … } ;
2 AAA* createAAA() 
3 {
4         static AAA a ;
5         return &a ;
6 }
7 


在調試狀態下,匯編代碼如下(請觀察藍色標記出來的代碼):
AAA* createAAA()
{
     …
     static AAA a ;

[1] 00401056 call        AAA::AAA (4010A0h)
[2] 0040105B push      offset `createAAA'::`2'::a::`dynamic atexit destructor' (442410h)
[3] 00401060 call        atexit (409A50h)
00401065 add            esp,4
00401068 mov           dword ptr [ebp-4],0FFFFFFFFh
     return &a ;
0040106F mov           eax,offset a (452620h)
}

00401091 ret  
注:[1]、[2]、[3]為方便說明加入的字符,實際代碼中并不存在。
[1]語句很明顯的調用AAA的構造函數。
[2]語句將442410h壓入棧中。
[3]語句調用atexit函數,根據我們的了解,atexit的參數應該是函數指針。那么我們來分析一下442410h處的代碼,從注釋來看,我們看到了destructor。代碼如下:
`createAAA'::`2'::a::`dynamic atexit destructor':

[1] 0044242E mov         ecx,offset a (452620h)
[2] 00442433 call        AAA::~AAA (403A90h)

0044244B ret           
[1]語句將a的地址放在ecx寄存器中,這是this調用規范的風格。
[2]語句調用AAA的析構函數。

程序結束時,將調用atexit函數注冊的442410h處的函數,進而調用了AAA的析構函數。從而保證了析構函數的調用。

 
3、   了解了局部靜態對象之后,我們來看看全局對象。
我們知道全局對象必須在main函數前已經被構造。為了弄清楚全局對象何時被構造,我在全局對象的實例化處設置了斷點,調用堆棧如下:
static.exe!aaaa::`dynamic initializer'() Line 22 C++
static.exe!_initterm(void (void)* * pfbegin=0x00451038, void (void)* * pfend=0x00451064) Line 707 C
static.exe!_cinit(int initFloatingPrecision=1) Line 208 + 0xf bytes C
static.exe!mainCRTStartup() Line 266 + 0x7 bytes C


作為對比,我在AAA的析構函數出設置了斷點,調用堆棧如下:
     static.exe!AAA::~AAA() Line 19  C++
     static.exe!aaaa::`dynamic atexit destructor'() + 0x28 bytes  C++
     static.exe!doexit(int code=0, int quick=0, int retcaller=0) Line 451  C
     static.exe!exit(int code=0) Line 311 + 0xd bytes  C
     static.exe!mainCRTStartup() Line 289  C


由此我們可以看出程序的實際入口點位mainCRTStartup而不是main函數(相對于ANSI的控制臺程序而言)。
我們來分析一下_cinit函數:
注釋中有一句[3. General C initializer routines],看來該函數的功能之一是完成C的初始化例程。
函數的核心代碼如下:
/*
         * do initializations
         */
        initret = _initterm_e( __xi_a, __xi_z );
/*
         * do C++ initializations
         */
        _initterm( __xc_a, __xc_z );
看來該函數主要進行C、C++的初始化。我們進一步分析函數_initterm_e和_initterm,兩個函數的功能進本相同,都是遍歷函數指針(由參數指定函數指針的開始位置[__xi_a、__xi_z]、結束位置[__xc_a、__xc_z]),如果函數指針不為null,那么調用該函數。
那么__xi_a、__xi_z和__xc_a、__xc_z到底代表了什么呢?在cinitexe.c文件中有如下代碼:
#pragma data_seg(".CRT$XIA")
_CRTALLOC(".CRT$XIA") _PVFV __xi_a[] = { NULL };
 
#pragma data_seg(".CRT$XIZ")
_CRTALLOC(".CRT$XIZ") _PVFV __xi_z[] = { NULL };/* C initializers */
 
#pragma data_seg(".CRT$XCA")
_CRTALLOC(".CRT$XCA") _PVFV __xc_a[] = { NULL };
 
#pragma data_seg(".CRT$XCZ")
_CRTALLOC(".CRT$XCZ") _PVFV __xc_z[] = { NULL };/* C++ initializers */
#pragma comment(linker, "/merge:.CRT=.data")
可以看出這四個變量分別在數據段.CRT$XIA、.CRT$XIZ、.CRT$XCA、.CRT$XCZ中。當連接器布局代碼時,它按根據的名稱,按照字母排序的規則,排列所有段。這樣在段.CRT$XIA中的變量出現在段.CRT$XIZ所有變量之前,從而形成鏈表。對于.CRT$XCA、.CRT$XCZ數據段同理。最后這四個數據段被合并到.data數據段中。
再看看這些變量的類型,typedef void (__cdecl *_PVFV)(void); 所以這些變量組成了2個初始化函數指針鏈表。
調試過程中,看到__xc_a、__xc_z鏈表中,指向的初始化函數很多是構造函數,如:
static std::_Init_locks initlocks;
static filebuf fout(_cpp_stdout);
extern _CRTDATA2 ostream cout(&fout);
cout對象也在此時被構造。
對于析構函數的調用也是采用相同的方式,只是此時每一種初始化,都有一種終止函數與之對應。

4、 總結
l         編譯、連接程序時,編譯器將所有全局對象的初始化函數放入.CRT$Xx中,連接器將所有的.CRT$XCx段合并成為.rdata數據段。在.CRT$XCA 到 .CRT$XCZ的所有段的數據組成初始化函數指針列表。
l   函數執行時,_initterm( __xc_a, __xc_z )函數調用所有的初始化函數。構造全局對象。構造對象完畢,調用atexit函數來保證析構函數的調用。Modern C++ Design就是通過控制調用atexit函數來決定對象的析構順序的。
l   對于靜態對象使用atexit來保證析構函數的調用。
l    程序結束時,調用exit來析構全局對象或靜態對象。


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/coffeemay/archive/2006/11/19/1395250.aspx

posted on 2011-04-29 11:06 沛沛 閱讀(896) 評論(0)  編輯 收藏 引用 所屬分類: C++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲一区影音先锋| 国产精品色婷婷| 亚洲精品一区二区三区婷婷月| 欧美日韩精品三区| 国产精品一国产精品k频道56| 9l国产精品久久久久麻豆| 男男成人高潮片免费网站| 久久国产精品一区二区三区| 国产欧美另类| 久久精品国产在热久久| 日韩一本二本av| 欧美日韩精品中文字幕| 一区二区日韩欧美| 一本综合精品| 国产精品s色| 午夜在线电影亚洲一区| 性欧美8khd高清极品| 国产欧美在线视频| 麻豆精品一区二区av白丝在线| 久久久欧美精品sm网站| 在线精品视频一区二区三四| 欧美成年网站| 久久精品亚洲国产奇米99| 在线精品视频一区二区三四| 亚洲高清视频在线观看| 欧美日韩一区二区在线播放| 久久久久久久91| 欧美日韩国产综合网| 久久久国产成人精品| 欧美激情网友自拍| 久久精品国产综合精品| 欧美美女bb生活片| 久久综合久久88| 国产精品久久久亚洲一区| 欧美国产日韩一区二区在线观看 | 欧美一区二区高清| 免费视频一区二区三区在线观看| 欧美亚洲综合另类| 欧美成人免费va影院高清| 久久精品五月| 国产精品久久久久91| 欧美国产精品久久| 黑人一区二区| 午夜一级在线看亚洲| 亚洲视频成人| 欧美激情第一页xxx| 久久综合九色九九| 国产一区久久| 亚洲在线第一页| 中文日韩电影网站| 欧美日韩p片| 亚洲欧洲一区二区三区久久| 在线日韩电影| 久久精品国产免费观看| 先锋影音国产一区| 国产精品高潮视频| 99视频精品| 亚洲一区二区三区免费观看| 欧美激情亚洲国产| 亚洲大片免费看| 亚洲国产婷婷香蕉久久久久久99| 久久久久久久久一区二区| 久久成人精品无人区| 国产美女精品免费电影| 亚洲天堂久久| 亚洲福利精品| 精品成人国产| 欧美伊人久久久久久久久影院 | 精品av久久久久电影| 欧美亚洲在线视频| 久久精品国产69国产精品亚洲| 国产精品夜夜夜一区二区三区尤| 亚洲天天影视| 久久久不卡网国产精品一区| 精品动漫av| 美女任你摸久久| 亚洲娇小video精品| 亚洲一级片在线看| 国产精品永久免费视频| 亚洲欧美日韩国产一区二区三区| 亚洲欧美在线看| 国产亚洲欧洲| 欧美18av| 中文一区二区| 久久精品一区二区| 亚洲电影欧美电影有声小说| 欧美高清视频在线播放| 一道本一区二区| 久久久久9999亚洲精品| 亚洲国产激情| 国产精品国产三级国产aⅴ浪潮| 亚洲综合成人在线| 美女精品国产| 亚洲一区三区在线观看| 国模精品娜娜一二三区| 久久综合给合| 一区二区三区黄色| 开心色5月久久精品| 一区二区三区视频在线| 国产视频在线一区二区| 欧美精品三区| 欧美在线影院| 一本一本久久a久久精品牛牛影视| 久久久亚洲欧洲日产国码αv| 亚洲精品欧洲精品| 国产日本精品| 欧美日韩国产综合在线| 久久综合色影院| 午夜久久99| 亚洲精品在线免费| 免费亚洲婷婷| 久久精品日韩| 亚洲欧美欧美一区二区三区| 亚洲韩国一区二区三区| 国产人成精品一区二区三| 欧美日韩国产专区| 欧美成年人网| 久久久久欧美精品| 亚洲视频自拍偷拍| 亚洲精品视频中文字幕| 亚洲电影毛片| 久久亚洲一区| 欧美综合二区| 亚洲欧美综合v| 一区二区三区产品免费精品久久75 | 国产精品久久777777毛茸茸| 欧美成年人视频网站| 久久久久国产免费免费| 欧美一区永久视频免费观看| 一本色道久久88综合亚洲精品ⅰ| 欧美激情第4页| 母乳一区在线观看| 另类激情亚洲| 91久久精品久久国产性色也91| 亚洲欧美精品一区| a91a精品视频在线观看| 亚洲人成7777| 亚洲国产精品欧美一二99| 免费成人黄色片| 久久在线视频在线| 久久久噜噜噜久久狠狠50岁| 久久久成人网| 美国十次成人| 欧美va天堂| 亚洲国产精品成人| 亚洲国产一区二区三区a毛片| 欧美sm视频| 亚洲日本久久| 亚洲经典在线| 99精品国产热久久91蜜凸| 日韩视频一区| 亚洲一级电影| 午夜性色一区二区三区免费视频| 欧美一区二区三区视频在线| 欧美一区二区三区电影在线观看| 欧美一级成年大片在线观看| 久久不射电影网| 老司机凹凸av亚洲导航| 欧美国产日韩视频| 欧美日韩天堂| 国产亚洲精品一区二区| 激情欧美一区| 亚洲美女诱惑| 午夜亚洲视频| 久久永久免费| 亚洲人成人99网站| 亚洲作爱视频| 欧美一区二视频| 欧美激情第一页xxx| 国产女主播一区二区| 亚洲丰满在线| 亚洲男人第一av网站| 久久久久在线观看| 亚洲人成网站777色婷婷| 一区二区日韩免费看| 久久9热精品视频| 欧美激情综合色| 国产欧美日韩激情| 亚洲人成在线免费观看| 香蕉久久夜色精品| 欧美成人免费在线观看| 亚洲一区免费| 欧美成年人在线观看| 国产欧美一区二区三区久久 | 欧美午夜一区二区福利视频| 国产一区二区精品丝袜| 日韩一级在线观看| 久久青草欧美一区二区三区| 亚洲精品社区| 久久综合久色欧美综合狠狠| 欧美性一二三区| 亚洲激情影视| 久久久久久久久久久成人| 亚洲伦理精品| 美国十次成人| 国产欧美日韩亚州综合| 亚洲一区二区在线视频| 欧美成人tv| 久久精品亚洲一区二区| 国产日韩专区在线|