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; // ... }