1、導出C函數以用于C或C++的項目
如果使用C語言編寫的DLL,希望從中導出函數給C或C++的模塊訪問,則應使用 __cplusplus 預處理器宏確定正在編譯的語言。如果是從C++語言模塊使用,則用C鏈接聲明這些函數。如果使用此技術并為DLL提供頭文件,則這些函數可以原封不動地由C和C++模塊使用。
以下代碼演示可由 C 和 C++ 客戶端應用程序使用的頭文件:
// MyCFuncs.h
#ifdef __cplusplus
extern "C" { // only need to export C interface if
// used by C++ source code
#endif
__declspec( dllimport ) void MyCFunc();
__declspec( dllimport ) void AnotherCFunc();
#ifdef __cplusplus
}
#endif
MyCFunc()和AnotherCFunc()為C語言DLL的導出函數。
如果需要將C函數鏈接到C++可執行文件,并且函數聲明頭文件沒有使用上面的技術,則在C++源文件中添加下列內容以防止編譯器修飾C函數名:
extern "C"
{
#include "MyCHeader.h"
}
該代碼告訴編譯器"MyCHeader.h"是C寫的,不要修飾頭文件中的C函數名,否則連接的時候會找不到。
2、導出 C++ 函數以用于C語言項目
如果在用C++編寫的DLL中有希望從C語言模塊訪問的函數,應使用C鏈接而不是C++鏈接來聲明這些函數。除非另外指定,C++編譯器使用C++類型安全命名約定(也稱作名稱修飾)和C++調用約定(使用此調用約定從C調用會很困難)。
若要指定 C 鏈接,請在DLL中為函數聲明指定 extern "C"。例如:
extern "C" __declspec( dllexport ) int MyFunc(long parm1);
在C語言的函數中是無法直接調用C++代碼的,如果要調用,可以做一個wrapper,例如call_Lib_CPPFunction,它的聲明和實現如下:
// wrapper function
extern "C" void call_Lib_CPPFunction(Lib* p, DataAttribute* dataAttribute)
{
p->daFun(dataAttribute);
}
// daFun才是我們C++代碼的實現
void Lib::daFun(DataAttribute* dataAttribute)
{
map<string, MMSINFO>::iterator it;
// ...
}