国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C++ > 正文

C++編寫DLL動態鏈接庫的步驟與實現方法

2020-01-26 14:28:57
字體:
來源:轉載
供稿:網友

本文實例講述了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)#endifClass 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)#endifClass 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"EXPORTSGetInstance = 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里面的方法了。

希望本文所述對大家VC++程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 定日县| 芜湖县| 正定县| 乐平市| 曲麻莱县| 巩义市| 神木县| 溧水县| 从化市| 英德市| 中宁县| 维西| 宁城县| 微山县| 铜陵市| 丰宁| 南木林县| 德阳市| 开封市| 当雄县| 安宁市| 柏乡县| 郓城县| 彰化县| 沅陵县| 元朗区| 宁国市| 汝南县| 织金县| 津市市| 宁陕县| 崇州市| 黔江区| 方山县| 建德市| 嘉禾县| 洪雅县| 册亨县| 大渡口区| 博客| 芜湖县|