使用 AFX_EXT_CLASS 導(dǎo)出和導(dǎo)入
擴(kuò)展 DLL 使用 AFX_EXT_CLASS 宏導(dǎo)出類;鏈接到擴(kuò)展 DLL 的可執(zhí)行文件使用該宏導(dǎo)入類。用于生成擴(kuò)展 DLL 的相同頭文件可通過 AFX_EXT_CLASS 宏與鏈接到 DLL 的可執(zhí)行文件一起使用。
在 DLL 的頭文件中,將 AFX_EXT_CLASS 關(guān)鍵字添加到類的聲明中,如下所示:
class AFX_EXT_CLASS CMyClass : public CDocument
{
// <body of class>
};
當(dāng)定義了預(yù)處理器符號(hào) _AFXDLL 和 _AFXEXT 時(shí),該宏被 MFC 定義為 __declspec(dllexport)。但當(dāng)定義了 _AFXDLL 而未定義 _AFXEXT 時(shí),該宏被定義為 __declspec(dllimport)。定義后,預(yù)處理器符號(hào) _AFXDLL 指示共享 MFC 版本正在由目標(biāo)可執(zhí)行文件(DLL 或應(yīng)用程序)使用。當(dāng) _AFXDLL 和 _AFXEXT 都定義了時(shí),這指示目標(biāo)可執(zhí)行文件是擴(kuò)展 DLL。
由于從擴(kuò)展 DLL 導(dǎo)出時(shí),AFX_EXT_CLASS 被定義為 __declspec(dllexport),因此可以導(dǎo)出整個(gè)類,而不必將該類的所有符號(hào)的修飾名放到 .def 文件中。此方法由 MFC 示例 DLLHUSK 使用。
雖然使用此方法可以避免創(chuàng)建 .def 文件和類的所有修飾名,但由于名稱可以按序號(hào)導(dǎo)出,因此創(chuàng)建 .def 文件的效率更高。若要使用 .def 文件導(dǎo)出方法,請(qǐng)將下列代碼放在頭文件的開頭和結(jié)尾處:
#undef AFX_DATA
#define AFX_DATA AFX_EXT_DATA
// <body of your header file>
#undef AFX_DATA
#define AFX_DATA
警告
導(dǎo)出內(nèi)聯(lián)函數(shù)時(shí)要小心,因?yàn)樗鼈冇锌赡軐?dǎo)致版本沖突。內(nèi)聯(lián)函數(shù)擴(kuò)展到應(yīng)用程序代碼中;因此,如果以后重寫內(nèi)聯(lián)函數(shù),除非重新編譯應(yīng)用程序本身,否則內(nèi)聯(lián)函數(shù)不會(huì)被更新。通常,不用重新生成使用 DLL 函數(shù)的應(yīng)用程序就可以更新 DLL 函數(shù)。
導(dǎo)出類中的個(gè)別成員
有時(shí),您可能希望導(dǎo)出類中的個(gè)別成員。例如,如果導(dǎo)出 CDialog 派生類,可能只需要導(dǎo)出構(gòu)造函數(shù)和 DoModal 調(diào)用。可以對(duì)需要導(dǎo)出的個(gè)別成員使用 AFX_EXT_CLASS。
例如:
class CExampleDialog : public CDialog
{
public:
AFX_EXT_CLASS CExampleDialog();
AFX_EXT_CLASS int DoModal();
...
// rest of class definition
...
};
您不再導(dǎo)出類的所有成員,但由于 MFC 宏的工作方式,您可能會(huì)遇到其他問題。幾個(gè) MFC 的 Helper 宏實(shí)際聲明或定義數(shù)據(jù)成員。因此,還必須從 DLL 導(dǎo)出這些數(shù)據(jù)成員。
例如,當(dāng)生成擴(kuò)展 DLL 時(shí),DECLARE_DYNAMIC 宏的定義如下:
#define DECLARE_DYNAMIC(class_name) \
protected: \
static CRuntimeClass* PASCAL _GetBaseClass(); \
public: \
static AFX_DATA CRuntimeClass class##class_name; \
virtual CRuntimeClass* GetRuntimeClass() const; \
以 static AFX_DATA 打頭的行聲明類的內(nèi)部靜態(tài)對(duì)象。若要正確導(dǎo)出該類并從客戶端可執(zhí)行文件訪問運(yùn)行時(shí)信息,必須導(dǎo)出此靜態(tài)對(duì)象。由于靜態(tài)對(duì)象是用 AFX_DATA 修飾符聲明的,因此只需在生成 DLL 時(shí)將 AFX_DATA 定義為 __declspec(dllexport),并在生成客戶端可執(zhí)行文件時(shí)將 AFX_DATA 定義為 __declspec(dllimport)。由于已經(jīng)以此方式定義了 AFX_EXT_CLASS,因此只需參考類定義,將 AFX_DATA 重定義為與 AFX_EXT_CLASS 相同。
例如:
#undef AFX_DATA
#define AFX_DATA AFX_EXT_CLASS
class CExampleView : public CView
{
DECLARE_DYNAMIC()
// ... class definition ...
};
#undef AFX_DATA
#define AFX_DATA
MFC 總是在其宏的內(nèi)部定義的數(shù)據(jù)項(xiàng)上使用 AFX_DATA 符號(hào),因此此技術(shù)適用于所有這類情況。例如,它適用于 DECLARE_MESSAGE_MAP。