DLL中導出函數的聲明有兩種方式:
一種方式是:在函數聲明中加上__declspec(dllexport);
另外一種方式是:采用模塊定義(.def)文件聲明,(.def)文件為鏈接器提供了有關被鏈接程序的導出、屬性及其他方面的信息。
方式一:在函數聲明中加上__declspec(dllexport)
/// 在動態鏈接庫程序中
/// 聲明動態鏈接庫(**.dll)的對外接口函數TestFuction
extern "C" __declspec(dllexport) int TestFuction(int nType,char *strPath,std::vector<string> &vecData)
{
////do anything here////
return 0;
}
/// 在外部希望調用動態鏈接庫的程序中
/// 加載動態鏈接庫(**.dll)並調用其對外接口TestFuction
void func()
{
//typedef與函數TestFuction類型相同的函數指針為TESTDLL
typedef int (_cdecl * TESTDLL)(int nType,char *strPath,std::vector<string> &vecData);
HINSTANCE hmod;
//加載動態鏈接庫**.dll
hmod =::LoadLibrary(_TEXT("dll相對路徑\\**.dll"));
if(NULL == hmod)
{
TRACE("加載**.dll失敗");
}
//定義一個與函數TestFuction類型相同的函數指針lpproc
TESTDLL lpproc;
//搜索**.dll中函數名為TestFuction的對外接口
lpproc = (TESTDLL)GetProcAddress (hmod,"TestFuction");
//如果搜索成功
if(NULL != lpproc)
{
int nType = 0;
char* strPath = "Data";
std::vector<string> vecData;
//通過函數指針lpproc調用**.dll的接口函數TestFuction
int nResult = (*lpproc)(nType,strPath,vecData);
}
//...
//在恰當的時候釋放動態鏈接庫**.dll
FreeLibrary(hmod);
}
方式二:采用模塊定義(.def)文件聲明
首先創建 一個DLL程序(DllTestDef)
在*.cpp中
int __stdcall Add(int numa, int numb)
{
return (numa + numb);
}
int __stdcall Sub(int numa, int numb)
{
return (numa - numb);
}
然后創建一個.def的文件,在里面加上
;DllTestDef.lib : 導出DLL函數
;作者:----
LIBRARY DllTestDef
EXPORTS
Add @ 1
Sub @ 2
最后創建一個測試程序:.cpp文件如下:
#include <iostream>
#include <windows.h>
using namespace std;
typedef int (__stdcall *FUN)(int, int);
HINSTANCE hInstance;
FUN fun;
int main()
{
hInstance = LoadLibrary("DLLTestDef.dll");
if(!hInstance)
cout << "Not Find this Dll" << endl;
fun = (FUN)GetProcAddress(hInstance, MAKEINTRESOURCE(1));
if (!fun)
{
cout << "not find this fun" << endl;
}
cout << fun(1, 2) << endl;
FreeLibrary(hInstance);
return 0;
}
說明:
.def文件的規則為:
(1)LIBRARY語句說明.def文件相應的DLL;
(2)EXPORTS語句后列出要導出函數的名稱。可以在.def文件中的導出函數名后加@n,表示要導出函數的序號為n(在進行函數調用時,這個序號將發揮其作用);
(3).def 文件中的注釋由每個注釋行開始處的分號 (;) 指定,且注釋不能與語句共享一行。
(4)使用__declspec(dllexport)和使用.def文件是有區別的。
如果你的DLL是提供給VC用戶使用的,你只需要把編譯DLL時產生的.lib提供給用戶,
它可以很輕松地調用你的DLL。但是如果你的DLL是供VB、PB、Delphi用戶使用的,那么會產生一個小麻煩。
因為VC++編譯器對於__declspec(dllexport)聲明的函數會進行名稱轉換,如下面的函數:
__declspec(dllexport) int __stdcall Add()
會轉換為Add@0,這樣你在VB中必須這樣聲明:
Declare Function Add Lib "DLLTestDef.dll" Alias "Add@0" () As Long
@后面的數由於參數類型不同而可能不同。這顯然不太方便。所以如果要想避免這種轉換,就要使用.def文件方式導出函數了。
http://www.cnblogs.com/enterBeijingThreetimes/archive/2010/08/04/1792099.html
1.VS2012創建.def文件 右鍵工程名->add->New Item->Visual C++->Code->Module-Define File(.def) 2.def文件的編寫 一個def文件必須有兩個部分:LIBRARY和EXPORTS 讓我們先看一個基本的.def文件稍后我將解析 LIBRARY dll_tutorial DESCRIPTION "our simple DLL" EXPORTS Add @1 Function @2 第一行,''LIBRARY''是一個必需的部分。它告訴鏈接器(linker)如何命名你的DLL。下面被標識為''DESCRIPTION''的部分並不是必需的,但是我喜歡把它放進去。該語句將字符串寫入 .rdata 節[據 MSDN],它告訴人們誰可能使用這個DLL,這個DLL做什么或它為了什么(存在)。再下面的部分標識為''EXPORTS''是另一個必需的部分;這個部分使得該函數可以被其它應用程序訪問到並且它創建一個導入庫。當你生成這個項目時,不僅是一個.dll文件被創建,而且一個文件擴展名為.lib的導出庫也被創建了。除了前面的部分以外,這里還有其它四個部分標識為:NAME, STACKSIZE, SECTIONS, 和 VERSION。我將不再在本文中涉及這些內容,但是如果你在Internet上搜索,我想你將找到一些東西(譯注: MSDN2003上對模板定義文件各部分內容有詳盡解釋,請參閱)。另外,一個分號(;)開始一個注解 3.def文件路徑的設置 Lingker->Input->Module Definition File 4.def文件的作用 通俗解釋: 在VC++中,生成DLL可以不使用.def文件。只需要在VC++的函數定義前要加 __declspec(dllexport)修飾就可以了。但是使用__declspec(dllexport)和使用.def文件是有區別的。如果 DLL是提供給VC++用戶使用的,你只需要把編譯DLL時產生的.lib提供給用戶,它可以很輕松地調用你的DLL。但是如果你的DLL是供其他程序如 VB、delphi,以及.NET 用戶使用的,那么會產生一個小麻煩。因為VC++對於 __declspec(dllexport)聲明的函數會進行名稱轉換,如下面的函數: __declspec(dllexport) int __stdcall IsWinNT() 會轉換為IsWinNT@0,這樣你在VB中必須這樣聲明: Declare Function IsWinNT Lib "my.dll" Alias "IsWinNT@0" () As Long @的后面的數由於參數類型不同而可能不同。這顯然不太方便。所以如果要想避免這種轉換,就要使用.def文件方式。 EXPORTS后面的數可以不給,系統會自動分配一個數。對於VB、PB、 Delphi用戶,通常使用按名稱進行調用的方式,這個數關系不大,但是對於使用.lib鏈接的VC程序來說,不是按名稱進行調用,而是按照這個數進行調用的,所以最好給出。 例子:我們用VC6.0制作一個dll,不使用.def文件,在頭文件中這樣寫 #ifndef LIB_H #define LIB_H extern "C" int _declspec(dllexport)add(int x,int y); #endif
http://blog.sina.com.cn/s/blog_78fd98af0101ghno.html
http://www.cnblogs.com/leijiangtao/p/4805940.html