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

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

  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 沛沛 閱讀(902) 評論(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>
            国产伦精品一区二区三区视频黑人| 欧美亚洲视频一区二区| 亚洲欧美视频在线观看视频| 欧美日韩三区四区| 一级日韩一区在线观看| 洋洋av久久久久久久一区| 国产精品美女主播在线观看纯欲| 性感少妇一区| 久久久综合香蕉尹人综合网| 亚洲精品欧美专区| 一本大道久久a久久综合婷婷| 亚洲成色777777女色窝| 亚洲日韩欧美视频| 亚洲高清视频中文字幕| 欧美精品一区二| 亚洲欧美国产精品va在线观看| 午夜在线成人av| 亚洲欧洲日本mm| 亚洲欧美国产高清| 亚洲国产高清视频| 亚洲综合欧美日韩| 亚洲伊人观看| 欧美国产极速在线| 久久国产综合精品| 欧美日韩高清在线| 久久一区免费| 欧美日韩在线观看视频| 一本色道久久综合亚洲二区三区| 一区二区三区在线观看国产| 一本大道av伊人久久综合| 一区二区高清在线观看| 久久全球大尺度高清视频| 亚洲免费人成在线视频观看| 欧美日韩亚洲免费| 亚洲性色视频| 日韩视频国产视频| 久久久久久999| 久久五月天婷婷| 在线精品一区二区| 欧美成人高清| 欧美成人精品激情在线观看| 国产欧美在线观看| 一道本一区二区| 亚洲欧美激情视频| 国产亚洲a∨片在线观看| 亚洲一区二区三区视频| 99精品视频免费观看视频| 久久香蕉国产线看观看av| 欧美激情一区二区三区在线视频| 国产精品爽黄69| 亚洲美女在线看| 亚洲精品欧美在线| 欧美偷拍一区二区| 欧美一区日韩一区| 国产精品xnxxcom| 香蕉久久国产| 欧美va亚洲va国产综合| 亚洲一区成人| 欧美日韩中文字幕在线| 欧美在线电影| 久久视频免费观看| 国产综合欧美在线看| 午夜精品一区二区三区四区 | 在线看国产日韩| 欧美激情2020午夜免费观看| 亚洲一区综合| 欧美激情影院| 欧美一区二区福利在线| 亚洲欧洲另类| 国产亚洲免费的视频看| 亚洲欧美国内爽妇网| 欧美黄色精品| 久久九九热re6这里有精品| 国产日韩在线播放| 欧美精品在欧美一区二区少妇| 91久久综合亚洲鲁鲁五月天| 日韩视频一区二区三区| 国内视频一区| 国产精品久久久久久超碰 | 久久精品卡一| 久久这里只有| 亚洲欧美日韩一区二区| 亚洲精品视频中文字幕| 国产一区二区三区久久久| 欧美高清你懂得| 久久国产精品久久久| 欧美高清视频免费观看| 日韩视频在线一区二区三区| 国产主播一区二区三区四区| 国产精品v日韩精品| 欧美ed2k| 亚洲网站视频福利| 亚洲精品视频免费观看| 欧美一区二区三区四区在线观看地址| 99亚洲伊人久久精品影院红桃| 黄色小说综合网站| 欧美精品激情在线观看| 久久综合网色—综合色88| 一本大道久久a久久精二百| 亚洲高清视频一区二区| 欧美激情一区二区三区| 奶水喷射视频一区| 久热爱精品视频线路一| 一本色道88久久加勒比精品| 亚洲国产欧美日韩另类综合| 国产精品欧美精品| 国产精品国产三级国产aⅴ9色| 欧美啪啪成人vr| 欧美电影免费| 欧美福利视频| 欧美精品激情| 欧美日韩国产另类不卡| 欧美日本中文| 欧美视频亚洲视频| 欧美视频在线一区| 国产精品视频内| 国产午夜精品美女毛片视频| 国产一区二区黄| 精品1区2区3区4区| 亚洲国产专区| 日韩亚洲欧美成人一区| 一区二区欧美日韩视频| 亚洲欧美在线免费观看| 亚洲麻豆av| 亚洲无线一线二线三线区别av| 亚洲天堂第二页| 亚洲精品在线免费观看视频| 亚洲美女av电影| 亚洲在线黄色| 久久久精品动漫| 亚洲国产精品精华液2区45| 久久手机免费观看| 欧美成人中文字幕| 亚洲精品美女| 亚洲欧美高清| 免费在线看一区| 欧美日韩一区二区免费在线观看 | 免费观看成人| 久久福利一区| 欧美亚洲综合在线| 久久综合一区二区| 亚洲激情女人| 午夜欧美精品| 欧美国产日韩一二三区| 国产精品久久97| 在线日韩中文| 亚洲一级影院| 欧美777四色影视在线| 在线亚洲自拍| 亚洲视频一区二区免费在线观看| 亚洲欧美一区在线| 美女在线一区二区| 久久综合色婷婷| 国产精品青草综合久久久久99| 一区在线播放视频| 亚洲欧美日韩国产中文在线| 久热国产精品| 亚洲视频免费在线观看| 美女被久久久| 国产主播一区二区三区| 亚洲一区二区免费视频| 欧美1区2区3区| 亚洲一区二区三区成人在线视频精品 | 久久久久国产一区二区三区四区| 亚洲国产小视频在线观看| 香蕉av福利精品导航| 欧美日韩在线精品| 亚洲黄色免费电影| 久久一区二区三区超碰国产精品| 亚洲深夜av| 欧美精品久久天天躁| 在线观看日韩av电影| 99ri日韩精品视频| 免费成人毛片| 一区二区三区四区五区视频| 久久免费视频网站| 亚洲欧美日韩中文在线制服| 欧美日韩色综合| 日韩视频在线免费观看| 欧美黄色一级视频| 久久天天躁狠狠躁夜夜爽蜜月| 国产欧美日韩在线观看| 亚洲欧美综合精品久久成人| 夜夜嗨av一区二区三区四季av| 欧美成人亚洲成人| 亚洲人成7777| 亚洲国产99精品国自产| 久久色中文字幕| 在线观看视频一区二区欧美日韩 | 久久久综合免费视频| 欧美亚洲三级| 国产一区二区精品在线观看| 久久精品视频在线看| 午夜一区不卡| 国产曰批免费观看久久久| 久久国产精品网站| 久久高清国产| 伊人狠狠色丁香综合尤物| 欧美大尺度在线| 欧美 日韩 国产在线|