__declspec(selectany)的做用

不懂就要問百度 html

轉載自 http://blog.163.com/cumt_xl/blog/static/19071504420127114610861/ c++

最 近在用 template 編寫singleton模式代碼的時候,遇到了一個問題,template要求實現要在同一個文件中,因此,我只能在h文件中定義並實現 singleton 模式類。類中必然要有靜態成員變量,靜態成員變量的定義成了問題,若是我放在cpp文件中,模板是不支持的,放在h文件中,若是h文件被屢次包含,會出現 重定義的狀況。 編程

    回來,請教高手,得知,能夠在初始化靜態成員變量前面加上__declspec(selectany) ,這樣編譯器會自動剔除對該靜態成員的重複定義。 app

   最近半年也一直用WTL,ATL,COM等。其實在WTL,ATL中已經大量使用了__declspec(selectany)方法。我猜測這是爲解決template單文件編程和靜態成員變量在頭文件中定義會出現重複定義矛盾而提出的。 函數

總的來講: htm

    __declspec(selelctany) 使在頭文件中定義靜態成員變量可行。 對象

=================================================================================== blog

其餘資料: get

selectany使用在c/c++工程的鏈接期間,通常用得不多,因此很陌生。 編譯器

這個屬性告訴編譯器聲明的全局變量是一個"任一揀選"(pick-any)COMDAT.在鏈接時間,若是多個COMDAT定義能看到,鏈接器選擇一個而且丟棄全部的剩餘的。若是鏈接器選項/OPT:REF被選擇,COMDAT中全部的沒有引用的數據項被刪除。

一個全局數據在EXE或者DLL中只能被初始化一次。當同一個頭文件被多個源文件引用時,在頭中定義全局數據始始化時,這個屬性被使用。這個屬性在c和c++的編譯器中都是可用的。


COMDAT record
一個經常使用對象文件格式(COFF)記錄,它包含的已被初始化的經常使用塊數據和打包的函數對鏈接器是能夠見的。
packaged function
當函數級的鏈接功能選擇開關被打開時,一個函數能被編譯器建立。在編譯器產生的對象文件中COMDAT記錄的打包的函數對於鏈接器是可見的。沒有打包的函數只在對象級(the object level)上鍊接。

下面是MSDN上一些例子:

//Correct - x1 is initialized and externally visible

__declspec(selectany) int x1=1;

//Incorrect - const is by default static in C++, so

//x2 is not visible externally (This is OK in C, since

//const is not by default static in C)

const __declspec(selectany) int x2 =2;

//Correct - x3 is extern const, so externally visible

extern const __declspec(selectany) int x3=3;

//Correct - x4 is extern const, so it is externally visible

extern const int x4;

const __declspec(selectany) int x4=4;

//Incorrect - __declspec(selectany) is applied to the uninitialized

//declaration of x5

extern __declspec(selectany) int x5;


轉自——http://liftstring.blog.sohu.com/137578737.html

相關文章
相關標籤/搜索