C++編寫DLL動(dòng)態(tài)鏈接庫的步驟與實(shí)現(xiàn)方法
本文實(shí)例講述了C++編寫DLL動(dòng)態(tài)鏈接庫的步驟與實(shí)現(xiàn)方法。分享給大家供大家參考,具體如下:
在寫C++程序時(shí),時(shí)常需要將一個(gè)class寫成DLL,供客戶端程序調(diào)用。這樣的DLL可以導(dǎo)出整個(gè)class,也可以導(dǎo)出這個(gè)class的某個(gè)方法。
一、導(dǎo)出整個(gè)class
方法很簡單,只需要在類的頭文件中class和類名之間加上_declspec(dllexport),同時(shí)在另外一份提供給客戶端調(diào)用程序使用的類的頭文件中class和類名之間加上_declspec(dllimport)。為了能讓客戶端程序和DLL程序公用該類的一份頭文件,通常在類的頭文件中使用宏和預(yù)編譯指令來處理。如下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編譯時(shí)DLL_TEST_API被定義為_declspec(dllexport),而且客戶端程序編譯時(shí)它被定義為_declspec(dllimport)。
二、導(dǎo)出這個(gè)類的某個(gè)或者某幾個(gè)方法
這時(shí),需要將_declspec(dllexport)放到成員函數(shù)名前,如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); };
但是,如果僅僅是這樣的話,當(dāng)客戶端程序#include這個(gè)頭文件后,定義DLLTest這個(gè)類的一個(gè)對(duì)象后(靜態(tài)方式鏈接DLL),客戶端程序無法鏈接通過,會(huì)提示構(gòu)造函數(shù)和析構(gòu)函數(shù)無法解析,此時(shí),需要將構(gòu)造函數(shù)和析構(gòu)函數(shù)前也加上DLL_TEST_API宏即可。
當(dāng)然這里還有個(gè)問題就是類的函數(shù)在導(dǎo)出后,名字會(huì)發(fā)生變化,我們可以在函數(shù)名前再加上extern "C" ,如 extern "C" DLL_TEST_API int Add(int a ,int b);但這只解決了C與C++調(diào)用時(shí)名字變更問題,可靠的方法還是增加一個(gè)模塊定義文件def,在該文件中定義導(dǎo)出函數(shù)的名稱,我們將在后面看到樣例。
DLL編寫完成后,就只剩下客戶端程序如何去調(diào)用該DLL了,靜態(tài)方式調(diào)用DLL和動(dòng)態(tài)方式調(diào)用DLL。
一、靜態(tài)方式調(diào)用DLL
這個(gè)方法就簡單了,將DLLTest.h頭文件和DLLTest.lib,DLLTest.dll文件拷貝到客戶端程序的當(dāng)前目錄下,在客戶端程序中#include<DLLTest.h>,然后通過#pragma comment(lib,"DLLTest.lib")的方式引入lib庫,或者在客戶端程序的工程屬性里面增加對(duì)該lib文件的引入。
然后就可以在客戶端程序中如同使用本地的一個(gè)class一樣使用該DLL了,如:
CDLLTest dllTest; dllTest.Add(1,2);
二、動(dòng)態(tài)方式調(diào)用DLL
動(dòng)態(tài)調(diào)用這個(gè)DLL,就需要對(duì)這個(gè)class進(jìn)行修改了。
首先,在DLLTest.cpp文件中增加一個(gè)全局函數(shù),該函數(shù)可以返回這個(gè)class的一個(gè)實(shí)例,這樣,客戶端程序調(diào)用這個(gè)全局函數(shù)后,得到該class的實(shí)例,就可以調(diào)用該class的實(shí)例方法了。
extern "C" _declspec(dllexport) CDLLTest* GetInstance() { return new CDLLTest; }
注:extern "C" 只是解決了c與c++編譯器之間的兼容問題,如果需要和其他編譯器之間兼容,可靠的辦法還是增加一個(gè).def文件,文件內(nèi)容如下:
LIBRARY "DLLTest" EXPORTS GetInstance = GetInstance
這樣就指定了DLL的函數(shù)導(dǎo)出后的名稱仍然不變。
這樣,客戶端程序就可以通過該函數(shù)來獲取class的一個(gè)實(shí)例了。如下:
先需要定義一個(gè)函數(shù)指針類型:
typedef CDllTestBase* (*pfGetInst)(); //注:CDllTestBase類后面會(huì)介紹。 HMOUDLE hMod = LoadLibrary( _T("DLLTest.DLL") ); if(hMod) { pfGetInst pfGetInstance = (pfGetInst)GetProcAddress("GetInstance"); if( p ) { //通過基類指針指向派生類對(duì)象 CDllTestBase * pInst = pfGetInstance (); if( NULL != pInst ) { pInst->Add( 1,2); } if( NULL != pInst ) { //釋放對(duì)象 delete pInst; } } }
當(dāng)然,這里還是需要include這個(gè)DLL的頭文件DLLTestBase.h,如果將之前所寫的頭文件DLLTest.h直接拷貝到客戶端程序的當(dāng)前目錄下,并include進(jìn)來的話,在編譯連接時(shí),是無法通過的,我們需要對(duì)這個(gè)頭文件進(jìn)行修改,首先增加一個(gè).h 文件DLLTestBase.h,在這個(gè)文件中我們將需要在客戶端程序中調(diào)用的函數(shù)都命名成純虛函數(shù),然后讓CDLLTest類繼承自CDLLTestBase類,DLLTestBase.h如下:
Class CDLLTestBase { Public: Virtual ~CDLLTestBase(){};//虛析構(gòu)函數(shù),且為內(nèi)聯(lián)函數(shù) 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需要提供一個(gè)虛析構(gòu)函數(shù),這樣在客戶端程序中就可以通過基類指針來釋放派生類對(duì)象了。
這樣,只需要將DLLTestBase.h拷貝到客戶端程序的當(dāng)前目錄下,然后在客戶端程序中#include"DLLTestBase.h",就可以如上面介紹一樣在客戶端程序中調(diào)用DLL里面的方法了。
希望本文所述對(duì)大家VC++程序設(shè)計(jì)有所幫助。
相關(guān)文章
C++實(shí)現(xiàn)LeetCode(8.字符串轉(zhuǎn)為整數(shù))
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(8.字符串轉(zhuǎn)為整數(shù)),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07C++中引用、內(nèi)聯(lián)函數(shù)、auto關(guān)鍵字和范圍for循環(huán)詳解
本文主要梳理了C++當(dāng)中一些瑣碎的知識(shí)點(diǎn),包括有命名空間,缺省參數(shù),引用,auto關(guān)鍵字和內(nèi)聯(lián)函數(shù),文中通過實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-02-02Qt?QtCreator添加自定義注釋的實(shí)現(xiàn)方法
在寫代碼的時(shí)候我們?yōu)榱艘?guī)范化,一般會(huì)加文件注釋、類注釋和函數(shù)注釋,本文主要介紹了Qt?QtCreator添加自定義注釋的實(shí)現(xiàn)方法,具有一定的參考價(jià)值,感興趣的可以了解一下2023-11-11哈希表實(shí)驗(yàn)C語言版實(shí)現(xiàn)
以下是對(duì)哈希表實(shí)驗(yàn)用C語言實(shí)現(xiàn)的代碼進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以參考下2013-07-07C++實(shí)現(xiàn)LeetCode(199.二叉樹的右側(cè)視圖)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(199.二叉樹的右側(cè)視圖),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08