c封裝dll文件夾
① 如何封裝c++程序成dll文件
C++編寫DLL的方法
在寫C++程序時,時常需要將一個class寫成DLL,供客戶端程序調用。這樣的DLL可以導出整個class,也可以導出這個class的某個方法。
一、導出整個class
方法很簡單,只需要在類的頭文件中class和類名之間加上_declspec(dllexport),同時在另外一份提供給客戶端調用程序使用的類的頭文件中class和類名之間加上_declspec(dllimport)。為了能讓客戶端程序和DLL程序公用該類的一份頭文件,通常在類的頭文件中使用宏和預編譯指令來處理。如下DLLTest.h:
#ifdef DLL_TEST_API
#else
#define
DLL_TEST_API _declspec(dllimport)
#endif
Class DLL_TEST_API CDLLTest
{
Public:
CDLLTest();
~CDLLTest();
int Add(
int
a, int
b);
};
DLLTest.cpp如下:
#define
DLL_TEST_API _declspec(dllexport)
#include 「DLLTest.h」
………………………………………
這樣,在DLL編譯時DLL_TEST_API被定義為_declspec(dllexport),而且客戶端程序編譯時它被定義為_declspec(dllimport)。
二、導出這個類的某個或者某幾個方法。
這時,需要將_declspec(dllexport)放到成員函數名前,如DLLTest.h:
#ifdef DLL_TEST_API
#else
#define
DLL_TEST_API _declspec(dllimport)
#endif
Class CDLLTest
{
Public:
CDLLTest();
~CDLLTest();
int DLL_TEST_API Add(
int
a, int
b);
};
但是,如果僅僅是這樣的話,當客戶端程序#include這個頭文件後,定義DLLTest這個類的一個對象後(靜態方式鏈接DLL),客戶端程序無法鏈接通過,會提示構造函數和析構函數無法解析,此時,需要將構造函數和析構函數前也加上DLL_TEST_API宏即可。
當然這里還有個問題就是類的函數在導出後,名字會發生變化,我們可以在函數名前再加上extern 「C」 ,如 extern 「C」 DLL_TEST_API int Add(int a ,int b);但這只解決了C與C++調用時名字變更問題,可靠的方法還是增加一個模塊定義文件def,在該文件中定義導出函數的名稱,我們將在後面看到樣例。
DLL編寫完成後,就只剩下客戶端程序如何去調用該DLL了,靜態方式調用DLL和動態方式調用DLL。
一、靜態方式調用DLL
這個方法就簡單了,將DLLTest.h頭文件和DLLTest.lib,DLLTest.dll文件拷貝到客戶端程序的當前目錄下,在客戶端程序中#include<DLLTest.h>,然後通過#pragma comment(lib,」DLLTest.lib」)的方式引入lib庫,或者在客戶端程序的工程屬性裡面增加對該lib文件的引入。
然後就可以在客戶端程序中如同使用本地的一個class一樣使用該DLL了,如:
CDLLTest dllTest;
dllTest.Add(
1
,2
);
二、動態方式調用DLL
動態調用這個DLL,就需要對這個class進行修改了。
首先,在DLLTest.cpp文件中增加一個全局函數,該函數可以返回這個class的一個實例,這樣,客戶端程序調用這個全局函數後,得到該class的實例,就可以調用該class的實例方法了。
extern
「C」 _declspec(dllexport) CDLLTest* GetInstance()
{
return
new
CDLLTest;
}
註:extern 「C」 只是解決了c與c++編譯器之間的兼容問題,如果需要和其他編譯器之間兼容,可靠的辦法還是增加一個.def文件,文件內容如下:
LIBRARY 「DLLTest」
EXPORTS
GetInstance
= GetInstance
這樣就指定了DLL的函數導出後的名稱仍然不變。
這樣,客戶端程序就可以通過該函數來獲取class的一個實例了。如下:
先需要定義一個函數指針類型:
typedef CDllTestBase* (*pfGetInst)();
//
註:CDllTestBase類後面會介紹。
HMOUDLE hMod
= LoadLibrary( _T(「DLLTest.DLL」) );
if(hMod)
{
pfGetInst pfGetInstance
= (pfGetInst)GetProcAddress(「GetInstance」);
if( p )
{
//
通過基類指針指向派生類對象
CDllTestBase * pInst = pfGetInstance ();
if( NULL
!= pInst )
{
pInst
->Add( 1
,2
);
}
if( NULL
!= pInst )
{
//
釋放對象
delete pInst;
}
}
}
當然,這里還是需要include這個DLL的頭文件DLLTestBase.h,如果將之前所寫的頭文件DLLTest.h直接拷貝到客戶端程序的當前目錄下,並include進來的話,在編譯連接時,是無法通過的,我們需要對這個頭文件進行修改,首先增加一個.h 文件DLLTestBase.h,在這個文件中我們將需要在客戶端程序中調用的函數都命名成純虛函數,然後讓CDLLTest類繼承自CDLLTestBase類,DLLTestBase.h如下:
Class CDLLTestBase
{
Public:
Virtual
~CDLLTestBase(){};//
虛析構函數,且為內聯函數
Virtual int
Add(int
a, int
b) = 0
;
}
DLLTest.h修改後如下:
#include 「DLLTestBase.h」
Class CDLLTest :
public
CDLLTestBase
{
Public:
CDLLTest();
~CDLLTest();
int
Add(int
a, int
b);
};
註:這里的DLLTestBase需要提供一個虛析構函數,這樣在客戶端程序中就可以通過基類指針來釋放派生類對象了。
這樣,只需要將DLLTestBase.h拷貝到客戶端程序的當前目錄下,然後在客戶端程序中#include」DLLTestBase.h」,就可以如上面介紹一樣在客戶端程序中調用DLL裡面的方法了。