解析在main函數(shù)之前調(diào)用函數(shù)以及對設(shè)計的作用詳解
更新時間:2013年05月23日 16:45:32 作者:
本篇文章是對在main函數(shù)之前調(diào)用函數(shù)以及對設(shè)計的作用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
前幾天為新員工寫一個簡單的測試框架,可讓他們方便的寫測試用例并且執(zhí)行。期間遇到一個問題就是如何讓他們增加測試用例而用不影響測試框架的代碼?c++的單件模式可以解決這個問題,但是其中一個難點是要在main之前注冊單件。c++可以通過構(gòu)造函數(shù)來實現(xiàn)注冊,c如何注冊?
最后查了下資料,原來可以定義在main之前調(diào)用的函數(shù)!有了這個特性可以改善c的模塊化設(shè)計。
特性介紹:
如果想定義在main函數(shù)之前調(diào)用的函數(shù),可以在函數(shù)的聲明之后加上一句“__attribute__((constructor))”,如下:
int before()__attribute__((constructor));
如果想定義在main函數(shù)之后調(diào)用的函數(shù),可以在函數(shù)的聲明之后加上一句“__attribute__((destructor))”,如下:
int after()__attribute__((destructor));
可以看得出來,應(yīng)該類似于c++中的構(gòu)造和析構(gòu)。
一些細(xì)節(jié)問題:
寫測試代碼測試了一下這個程序,發(fā)現(xiàn)幾點:
1、before在main之前調(diào)用,調(diào)用之前,各個全局變量已經(jīng)完成初始化。也就是說,這些函數(shù)是在全局變量初始化之后,main函數(shù)之前調(diào)用的。這一點是非常重要的,否則可能會引起很多的問題。
2、after在main之后調(diào)用,但是有一點比較特殊,必須是在main中return的話才執(zhí)行,否則,需要通過atexit執(zhí)行某函數(shù)。這個特性目前對我沒有太大的用處。
3、在main函數(shù)之前調(diào)用的函數(shù)可以聲明為static。
4、在main函數(shù)之前調(diào)用的函數(shù)可以調(diào)用多個。這里就有一個問題,就是這些函數(shù)的調(diào)用順序的問題。這個問題首先是一個設(shè)計的問題,也就是,我們應(yīng)該設(shè)計這些函數(shù)為順序無關(guān)的函數(shù)。另外,調(diào)用順序和編譯的順序相關(guān),我在linux下使用make進(jìn)行編譯,發(fā)現(xiàn)最后編譯的源文件中的函數(shù)會最先調(diào)用。
5、可以在庫(動態(tài)庫和靜態(tài)庫)中定義這樣的函數(shù)。
用對設(shè)計的作用:
1、可以優(yōu)化c++中的單件模式。參考《設(shè)計模式》
單件模式有一個最大的特點就是可以在運行過程中連接單件。如果使用條件語句來決定使用哪個單件硬性限定了可能的單件集合。所以,書中引入了一個單件注冊表的概念,書中對單件注冊表的初始化采用的是如下的做法:
首先定義一個單件類,在單件類的構(gòu)造函數(shù)中調(diào)用單件的注冊函數(shù)注冊自身:
MySingleton::MySingleton()
{
...
Singleton::Register("MySingleton", this);
}
這個函數(shù)是怎么被調(diào)用的那?可以定義一個靜態(tài)實例:
static MySingleton theSingleton;
這樣就會在main函數(shù)之前調(diào)用MySingleton的構(gòu)造函數(shù)來構(gòu)造這個靜態(tài)實例,從而達(dá)到像注冊表注冊的目的。
這個方案有個缺點:它能夠成功存在一個前提,就是在theSingleton實例化之前,單件注冊表列表必須存在,否則會失敗。則其實只是一個可能失敗的點,如果MySingleton還應(yīng)用其他的全局變量,則可能這個時候這些全局變量還沒有初始化。
解決這個問題的一個方案就是將單件注冊的時間由構(gòu)造函數(shù)移到main函數(shù)之前調(diào)用的函數(shù)中來。
定義函數(shù):
static void before_main()
{
Singleton::Register("MySingleton", &theSingleton);
}
聲明:
staitc void before_main()__attribute__((constructor));
before_main會在main函數(shù)之前調(diào)用,而調(diào)用時全局變量已經(jīng)全部初始化,這樣就可以避免上面的問題。
其實單件不單單可以在c++(面向?qū)ο螅┲惺褂?,也可以在c中使用。而且有了c的這個特性后,單件更好用。
2、構(gòu)造插件開發(fā)框架,而不用對框架進(jìn)行更改。
構(gòu)造插件開發(fā)框架的一個問題是:如何新增一個插件而不用修改主框架代碼就可以調(diào)用插件代碼。一般情況下都會使用插件注冊機制。也就是框架對外提供注冊接口,插件使用這些接口進(jìn)行注冊。c要實行此功能,一個可行的方案是在插件中定義main之前執(zhí)行的函數(shù),在此函數(shù)中調(diào)用插件注冊接口完成注冊。(注:這里討論的是插件的靜態(tài)加載)。
3、一個模塊有一些初始化工作要做,使用這種機制可以不更改main或者函數(shù)。
拋開插件框架,使用這個特性也可以對c的模塊化進(jìn)行很多優(yōu)化。比如,可以把各個模塊的初始化工作放在main之前進(jìn)行從而防止對main的頻繁修改。
注:本文描述的環(huán)境為linux c,c++。
最后查了下資料,原來可以定義在main之前調(diào)用的函數(shù)!有了這個特性可以改善c的模塊化設(shè)計。
特性介紹:
如果想定義在main函數(shù)之前調(diào)用的函數(shù),可以在函數(shù)的聲明之后加上一句“__attribute__((constructor))”,如下:
int before()__attribute__((constructor));
如果想定義在main函數(shù)之后調(diào)用的函數(shù),可以在函數(shù)的聲明之后加上一句“__attribute__((destructor))”,如下:
int after()__attribute__((destructor));
可以看得出來,應(yīng)該類似于c++中的構(gòu)造和析構(gòu)。
一些細(xì)節(jié)問題:
寫測試代碼測試了一下這個程序,發(fā)現(xiàn)幾點:
1、before在main之前調(diào)用,調(diào)用之前,各個全局變量已經(jīng)完成初始化。也就是說,這些函數(shù)是在全局變量初始化之后,main函數(shù)之前調(diào)用的。這一點是非常重要的,否則可能會引起很多的問題。
2、after在main之后調(diào)用,但是有一點比較特殊,必須是在main中return的話才執(zhí)行,否則,需要通過atexit執(zhí)行某函數(shù)。這個特性目前對我沒有太大的用處。
3、在main函數(shù)之前調(diào)用的函數(shù)可以聲明為static。
4、在main函數(shù)之前調(diào)用的函數(shù)可以調(diào)用多個。這里就有一個問題,就是這些函數(shù)的調(diào)用順序的問題。這個問題首先是一個設(shè)計的問題,也就是,我們應(yīng)該設(shè)計這些函數(shù)為順序無關(guān)的函數(shù)。另外,調(diào)用順序和編譯的順序相關(guān),我在linux下使用make進(jìn)行編譯,發(fā)現(xiàn)最后編譯的源文件中的函數(shù)會最先調(diào)用。
5、可以在庫(動態(tài)庫和靜態(tài)庫)中定義這樣的函數(shù)。
用對設(shè)計的作用:
1、可以優(yōu)化c++中的單件模式。參考《設(shè)計模式》
單件模式有一個最大的特點就是可以在運行過程中連接單件。如果使用條件語句來決定使用哪個單件硬性限定了可能的單件集合。所以,書中引入了一個單件注冊表的概念,書中對單件注冊表的初始化采用的是如下的做法:
首先定義一個單件類,在單件類的構(gòu)造函數(shù)中調(diào)用單件的注冊函數(shù)注冊自身:
復(fù)制代碼 代碼如下:
MySingleton::MySingleton()
{
...
Singleton::Register("MySingleton", this);
}
這個函數(shù)是怎么被調(diào)用的那?可以定義一個靜態(tài)實例:
static MySingleton theSingleton;
這樣就會在main函數(shù)之前調(diào)用MySingleton的構(gòu)造函數(shù)來構(gòu)造這個靜態(tài)實例,從而達(dá)到像注冊表注冊的目的。
這個方案有個缺點:它能夠成功存在一個前提,就是在theSingleton實例化之前,單件注冊表列表必須存在,否則會失敗。則其實只是一個可能失敗的點,如果MySingleton還應(yīng)用其他的全局變量,則可能這個時候這些全局變量還沒有初始化。
解決這個問題的一個方案就是將單件注冊的時間由構(gòu)造函數(shù)移到main函數(shù)之前調(diào)用的函數(shù)中來。
定義函數(shù):
復(fù)制代碼 代碼如下:
static void before_main()
{
Singleton::Register("MySingleton", &theSingleton);
}
聲明:
staitc void before_main()__attribute__((constructor));
before_main會在main函數(shù)之前調(diào)用,而調(diào)用時全局變量已經(jīng)全部初始化,這樣就可以避免上面的問題。
其實單件不單單可以在c++(面向?qū)ο螅┲惺褂?,也可以在c中使用。而且有了c的這個特性后,單件更好用。
2、構(gòu)造插件開發(fā)框架,而不用對框架進(jìn)行更改。
構(gòu)造插件開發(fā)框架的一個問題是:如何新增一個插件而不用修改主框架代碼就可以調(diào)用插件代碼。一般情況下都會使用插件注冊機制。也就是框架對外提供注冊接口,插件使用這些接口進(jìn)行注冊。c要實行此功能,一個可行的方案是在插件中定義main之前執(zhí)行的函數(shù),在此函數(shù)中調(diào)用插件注冊接口完成注冊。(注:這里討論的是插件的靜態(tài)加載)。
3、一個模塊有一些初始化工作要做,使用這種機制可以不更改main或者函數(shù)。
拋開插件框架,使用這個特性也可以對c的模塊化進(jìn)行很多優(yōu)化。比如,可以把各個模塊的初始化工作放在main之前進(jìn)行從而防止對main的頻繁修改。
注:本文描述的環(huán)境為linux c,c++。
您可能感興趣的文章:
- JavaScript學(xué)習(xí)筆記(三):JavaScript也有入口Main函數(shù)
- C++改變編程入口為main函數(shù)
- C語言之沒有main函數(shù)的helloworld示例
- c語言main函數(shù)使用及其參數(shù)介紹
- C語言main函數(shù)的參數(shù)及其返回值詳細(xì)解析
- C語言中怎么在main函數(shù)開始前執(zhí)行函數(shù)
- 深入探討:main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼?
- 深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- C#難點逐個擊破(4):main函數(shù)
- .NET中函數(shù)Main的使用技巧
相關(guān)文章
通過c++的sort函數(shù)實現(xiàn)成績排序功能
這篇文章主要介紹了通過c++的sort函數(shù)實現(xiàn)成績排序,本文通過實例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2020-02-02