#ifdef __cplusplus extern "C" { //函數(shù)聲明 //變量聲明,變量一般前面都有extern //類聲明,這個不起作用,編譯器直接忽略掉class前面的extern “C” #ifdef __cplusplus } #endif
__declspec(dllexport) __declspec(dllimport)一般也是使用宏的形式: #ifdef ONEDLL_EXPORTS #define ONEDLL_API __declspec(dllexport) #else #define ONEDLL_API __declspec(dllimport) #endif上面的兩個宏結(jié)合一下就是下面這樣的了: // 下列 ifdef 塊是創(chuàng)建使從 DLL 導(dǎo)出更簡單的 // 宏的標(biāo)準(zhǔn)方法。此 DLL 中的所有文件都是用命令行上定義的 ONEDLL_EXPORTS // 符號編譯的。在使用此 DLL 的 // 任何其他項目上不應(yīng)定義此符號。這樣,源文件中包含此文件的任何其他項目都會將 // ONEDLL_API 函數(shù)視為是從 DLL 導(dǎo)入的,而此 DLL 則將用此宏定義的 // 符號視為是被導(dǎo)出的。 #ifdef ONEDLL_EXPORTS #define ONEDLL_API __declspec(dllexport) #else #define ONEDLL_API __declspec(dllimport) #endif![]() // 此類是從 OneDll.dll 導(dǎo)出的 #ifdef __cplusplus extern "C" { #endif class ONEDLL_API COneDll { public: COneDll(void); ~COneDll(void); // TODO: 在此添加您的方法。 int m_a; int m_b; int *m_p; int m_n;![]() void AddValue();![]() };![]() extern ONEDLL_API int nOneDll;![]() ONEDLL_API int fnOneDll(void);![]() #ifdef __cplusplus } #endif![]() 如果調(diào)用模塊和被調(diào)用模塊都是C++(而且是同一種編成環(huán)境,如VC,甚至需要同一版本的VC),那么就不需要extern “C”了,因為這個標(biāo)志的作用就是用在函數(shù)和變量聲明前,無論是調(diào)用模塊,還是被調(diào)用模塊,都將生成C修飾符,調(diào)用模塊將需要C修飾符的函數(shù),而被調(diào)用模塊將產(chǎn)生C修飾符的函數(shù),所以這個標(biāo)志在兩者都是C++的時候使用并不受影響,不使用這個標(biāo)志,也不受影響。 但是如果C模塊要調(diào)用C++ 模塊,那么C++模塊就需要使用extern “C”,當(dāng)然C不用,由于是在頭文件的聲明中使用,所以使用下面的宏能夠使得這個頭文件也在C中順利使用: #ifdef __cplusplus extern "C" { //函數(shù)聲明 //變量聲明,變量一般前面都有extern //類聲明,這個不起作用,編譯器直接忽略掉class前面的extern “C” #ifdef __cplusplus } #endif如果C++模塊要調(diào)用C模塊,那么C++模塊還是需要extern “C”,當(dāng)然C不用,由于是在頭文件的聲明中使用,所以使用上面的宏同樣能夠使得這個頭文件也在C中順利使用。 總結(jié)一下就是加上extern “C”在什么情況下都沒錯,但是要注意函數(shù)重載的問題。 def文件是一種比較麻煩的方法,下面是MSDN中的部分內(nèi)容: 模塊定義 (.def) 文件是包含一個或多個描述 DLL 各種屬性的 Module 語句的文本文件。如果不使用 __declspec(dllexport) 關(guān)鍵字導(dǎo)出 DLL 的函數(shù),則 DLL 需要 .def 文件。 .def 文件必須至少包含下列模塊定義語句: LIBRARY BTREE EXPORTS Insert @1 Delete @2 Member @3 Min @4![]()
其實__declspec(dllexport)的作用就是讓編譯器按照某種預(yù)定的方式(前面大致解釋了這種方式的規(guī)則)來輸出導(dǎo)出函數(shù)及變量的符號,而def文件則是自己為每一個函數(shù)和變量指定導(dǎo)出符號,所以def是一個非自動化,手工很強的方式,不是特殊情況的話,實在沒有必要浪費這些時間。 還有一個問題,就是使用def會把調(diào)用方式和__declspec(dllexport)的作用全部覆蓋掉,所以還需要自己處理調(diào)用方式不同產(chǎn)生的錯誤。 一般使用def文件的情況是你需要使用運行時加載,并且需要使用GetProcAddress函數(shù)獲得函數(shù)地址,這個函數(shù)需要直接指明函數(shù)產(chǎn)生的導(dǎo)出符號,而可以自己指定導(dǎo)出符號的方式就是使用def。 def文件的具體語法可以看看msdn。 |
|
|