使用DLL的一個(gè)比較嚴重的問(wèn)題就是編譯器之間的兼容性問(wèn)題。不同的編譯器對c++函數在二進(jìn)制級別的實(shí)現方式是不同的。所以對基于
C++的DLL,如果編譯器不同就有很麻煩的。如果創(chuàng )建的是MFC擴展DLL,就不會(huì )存在問(wèn)題,因為它只能被動(dòng)態(tài)連接到MFC的客戶(hù)應用程序。這里不是本文討論的重點(diǎn)。
一、重新編譯問(wèn)題
我們先來(lái)看一個(gè)在實(shí)際中可能遇到的問(wèn)題:
比如現在建立好了一個(gè)DLL導出了CMyClass類(lèi),客戶(hù)也能正常使用這個(gè)DLL,假設CMyClass對象的大小為30字節。如果我們需要修改DLL中的CMyClass類(lèi),讓它有相同的函數和成員變量,但是給增加了一個(gè)私有的成員變量int類(lèi)型,現在CMyClass對象的大小就是34字節了。當直接把這個(gè)新的DLL給客戶(hù)使用替換掉原來(lái)30字節大小的DLL,客戶(hù)應用程序期望的是30字節大小的對象,而現在卻變成了一個(gè)34字節大小的對象,糟糕,客戶(hù)程序出錯了。
類(lèi)似的問(wèn)題,如果不是導出CMyClass類(lèi),而在導出的函數中使用了CMyClass,改變對象的大小仍然會(huì )有問(wèn)題的。這個(gè)時(shí)候修改這個(gè)問(wèn)題的唯一辦法就是替換客戶(hù)程序中的CMyClass的頭文件,全部重新編譯整個(gè)應用程序,讓客戶(hù)程序使用大小為34字節的對象。
這就是一個(gè)嚴重的問(wèn)題,有的時(shí)候如果沒(méi)有客戶(hù)程序的源代碼,那么我們就不能使用這個(gè)新的DLL了。
二、解決方法 為了能避免重新編譯客戶(hù)程序,這里介紹兩個(gè)方法:(1)使用接口類(lèi)。(2)使用創(chuàng )建和銷(xiāo)毀類(lèi)的靜態(tài)函數。
1、使用接口類(lèi)
接口類(lèi)的也就是創(chuàng )建第二個(gè)類(lèi),它作為要導出類(lèi)的接口,所以在導出類(lèi)改變時(shí),也不需要重新編譯客戶(hù)程序,因為接口類(lèi)沒(méi)有發(fā)生變化。
假設導出的CMyClass類(lèi)有兩個(gè)函數FunctionA FunctionB?,F在創(chuàng )建一個(gè)接口類(lèi)CMyInterface,下面就是在DLL中的CMyInterface類(lèi)的頭文件的代碼:
# include "MyClass.h"
class _declspec(dllexport) CMyInterface
{
CMyClass *pmyclass;
CMyInterface();
~CMyInterface();
public:
int FunctionA(int);
int FunctionB(int);
};
而在客戶(hù)程序中的頭文件稍不同,不需要INCLUDE語(yǔ)句,因為客戶(hù)程序沒(méi)有它的拷貝。相反,使用一個(gè)CMyClass的向前聲明,即使沒(méi)有頭文件也能編譯:
class _declspec(dllexport) CMyInterface
{
class CMyClass;//向前聲明
CMyClass *pmyclass;
CMyInterface();
~CMyInterface();
public:
int FunctionA(int);
int FunctionB(int);
};
在DLL中的CMyInterface的實(shí)現如下:
CMyInterface::CMyInterface()
{
pmyclass = new CMyClass();
}
CMyInterface::~CMyInterface()
{
delete pmyclass;
}
int CMyInterface::FunctionA()
{
return pmyclass->FunctionA();
}
int CMyInterface::FunctionB()
{
return pmyclass->FunctionB();
}
.....
對導出類(lèi)CMyClass的每個(gè)成員函數,CMyInterface類(lèi)都提供自己的對應的函數??蛻?hù)程序與CMyClass沒(méi)有聯(lián)系,這樣任意改CMyClass也不會(huì )有問(wèn)題,因為CMyInterface類(lèi)的大小沒(méi)有發(fā)生變化。即使為了能訪(fǎng)問(wèn)CMyClass中的新增變量而給CMyInterface類(lèi)加了函數也不會(huì )有問(wèn)題的。
但是這種方法也存在明顯的問(wèn)題,對導出類(lèi)的每個(gè)函數和成員變量都要對應實(shí)現,有的時(shí)候這個(gè)接口類(lèi)會(huì )很龐大。同時(shí)增加了客戶(hù)程序調用所需要的時(shí)間。增加了程序的開(kāi)銷(xiāo)
2、使用靜態(tài)函數
還可以使用靜態(tài)函數來(lái)創(chuàng )建和銷(xiāo)毀類(lèi)對象。創(chuàng )建一個(gè)導出類(lèi)的時(shí)候,增加兩個(gè)靜態(tài)的公有函數CreateMe()/DestroyMe(),頭文件如下:
class _declspec(dllexport) CMyClass
{
CMyClass();
~CMyClass();
public:
static CMyClass *CreateMe();
static void DestroyMe(CMyClass *ptr);
};
實(shí)現函數就是:
CMyClass * CMyClass::CMyClass()
{
return new CMyClass;
}
void CMyClass::DestroyMe(CMyClass *ptr)
{
delete ptr;
}
然后象其他類(lèi)一樣導出CMyClass類(lèi),這個(gè)時(shí)候在客戶(hù)程序中使用這個(gè)類(lèi)的方法稍有不同了。如若想創(chuàng )建一個(gè)CMyClass對象,就應該是:
CMyClass x;
CMyClass *ptr = CMyClass::CreateMe();
在使用完后刪除:
CMyClass::DestroyMe(ptr);