原諒轉(zhuǎn)載自:http://lukas06.blog.sohu.com/94010246.html
C++編譯器在生成DLL時,會對導(dǎo)出函數(shù)進(jìn)行名字改編,并且不同的編譯器使用的改編規(guī)則不一樣,因此改編的名字后的名字是不一樣的。因此,如果利用不同的分別生成DLL文件和訪問DLL文件的客戶端,那么后者在訪問該DLL文件的時候就會出現(xiàn)問題。例如:使用C++編寫了一個DLL,而使用C語言編寫的客戶端進(jìn)行訪問就會出現(xiàn)問題。由于C++編譯器已經(jīng)對該導(dǎo)出函數(shù)名字進(jìn)行了改編,所以用C語言編寫的客戶端就找不到DLL的導(dǎo)出函數(shù)。這就是DLL導(dǎo)出函數(shù)的名字改編問題。
如果希望動態(tài)鏈接庫文件在編譯時,導(dǎo)出函數(shù)的名稱不要發(fā)生改變,那么在定義導(dǎo)出函數(shù)時,需要加上限定符:extern "C"。注意:雙引號中的“C”一定要大寫。
例如:Dll1.h頭文件
#ifdef DLL1_API
#else
#define DLL1_API extern "C" _declspec(dllimport)
#endif
DLL1_API int add(int a,int b);
DLL1_API int subtract(int a,int b);
Dll1.cpp源文件
#define DLL1_API extern "C" _declspec(dllexport)
#include "Dll1.h"
int add(int a,int b)
{
return a+b;
}
int subtract(int a,int b)
{
return a-b;
}
這樣利用dumpbin工具可以查看Dll2.dll的導(dǎo)出函數(shù),>dumpbin -exports Dll2.dll,可以發(fā)現(xiàn)名字沒有被改編。
利用限定符extern "C"可以解決C++和C語言之間相互調(diào)用時函數(shù)命名的問題。但是這種方法有一個缺陷:就是不能用于導(dǎo)出一個類的成員函數(shù),而只能用于導(dǎo)出全局函數(shù)這種情況。
但是還有一個問題是,如果使用了標(biāo)準(zhǔn)調(diào)用約定,也就是pascal調(diào)用約定,WINAPI調(diào)用約定:_stdcall,此時即使使用了extern "C",仍然會出現(xiàn)導(dǎo)出函數(shù)名字被改編的問題。例如:使用C語言編寫一個DLL文件,而客戶端使用Delphi進(jìn)行編寫,那么在編寫導(dǎo)出函數(shù)時,應(yīng)該指定其使用標(biāo)準(zhǔn)的函數(shù)調(diào)用約定。此時,就會出現(xiàn)問題,即C語言編寫的DLL文件的導(dǎo)出函數(shù)發(fā)生了名字改編。在這種情況下,可以使用一個稱為模塊定義文件(DEF)的方式解決名字改編問題。
例如:Dll2.def
LIBRARY
EXPORTS
add
subtract
如果想使用與源文件中定義的不一樣的函數(shù)名,可以按照以下語法:entryname=internalname
其中,entryname是要導(dǎo)出的符號名,而internalname是DLL中將要導(dǎo)出的函數(shù)名。
示例:
LIBRARY "VerifyLocalResType"
EXPORTS
VerifyDDSSize
VerifyDDSType
VerifyTGASize
VerifyTGAType