?? mfc7.php
字號:
<HTML><HEAD><TITLE>MFC教程_ MFC的DLL</TITLE>
<META http-equiv=Content-Type content="text/html; charset=gb2312">
<META content="MSHTML 6.00.2800.1458" name=GENERATOR></HEAD>
<BODY bgColor=#ffffff>
<OL start=7>
<P align=justify>
<LI><A name=_Toc445889069></A><A name=_Toc445782472></A><A
name=_Toc452640933></A><A name=_Toc457299032></A><B>MFC的DLL</B>
<P></P>
<P
align=justify>一般的,在介紹Windows編程的書中講述DLL的有關(guān)知識較多,而介紹MFC的書則比較少地提到。即使使用MFC來編寫動態(tài)鏈接庫,對于初步接觸DLL的程序員來說,了解DLL的背景知識是必要的。另外,MFC提供了新的手段來幫助編寫DLL程序。所以,本節(jié)先簡潔的介紹有關(guān)概念。</P>
<OL>
<P align=justify>
<LI><A name=_Toc445889070></A><A name=_Toc445782473></A><A
name=_Toc452640934></A><A name=_Toc457299033></A><B>DLL的背景知識</B>
<P></P></LI></OL></LI></OL>
<OL>
<P align=justify>
<LI>靜態(tài)鏈接和動態(tài)鏈接
<P></P></LI></OL>
<P align=justify>當(dāng)前鏈接的目標(biāo)代碼(.obj)如果引用了一個函數(shù)卻沒有定義它,鏈接程序可能通過兩種途徑來解決這種從外部對該函數(shù)的引用:</P>
<UL>
<P align=justify>
<LI>靜態(tài)鏈接
<P></P></LI></UL>
<P
align=justify>鏈接程序搜索一個或者多個庫文件(標(biāo)準(zhǔn)庫.lib),直到在某個庫中找到了含有所引用函數(shù)的對象模塊,然后鏈接程序把這個對象模塊拷貝到結(jié)果可執(zhí)行文件(.exe)中。鏈接程序維護對該函數(shù)的所有引用,使它們指向該程序中現(xiàn)在含有該函數(shù)拷貝的地方。</P>
<UL>
<P align=justify>
<LI>動態(tài)鏈接
<P></P></LI></UL>
<P
align=justify>鏈接程序也是搜索一個或者多個庫文件(輸入庫.lib),當(dāng)在某個庫中找到了所引用函數(shù)的輸入記錄時,便把輸入記錄拷貝到結(jié)果可執(zhí)行文件中,產(chǎn)生一次對該函數(shù)的動態(tài)鏈接。這里,輸入記錄不包含函數(shù)的代碼或者數(shù)據(jù),而是指定一個包含該函數(shù)代碼以及該函數(shù)的順序號或函數(shù)名的動態(tài)鏈接庫。</P>
<P
align=justify>當(dāng)程序運行時,Windows裝入程序,并尋找文件中出現(xiàn)的任意動態(tài)鏈接。對于每個動態(tài)鏈接,Windows裝入指定的DLL并且把它映射到調(diào)用進程的虛擬地址空間(如果沒有映射的話)。因此,調(diào)用和目標(biāo)函數(shù)之間的實際鏈接不是在鏈接應(yīng)用程序時一次完成的(靜態(tài)),相反,是運行該程序時由Windows完成的(動態(tài))。</P>
<P align=justify>這種動態(tài)鏈接稱為加載時動態(tài)鏈接。還有一種動態(tài)鏈接方式下面會談到。</P>
<OL>
<P align=justify>
<LI>動態(tài)鏈接的方法
<P></P></LI></OL>
<P align=justify>鏈接動態(tài)鏈接庫里的函數(shù)的方法如下:</P>
<UL>
<P align=justify>
<LI>加載時動態(tài)鏈接(Load_time dynamic linking)
<P></P></LI></UL>
<P align=justify>如上所述。Windows搜索要裝入的DLL時,按以下順序:</P>
<P align=justify>應(yīng)用程序所在目錄→當(dāng)前目錄→Windows SYSTEM目錄→Windows目錄→PATH環(huán)境變量指定的路徑。</P>
<UL>
<P align=justify>
<LI>運行時動態(tài)鏈接(Run_time dynamic linking)
<P></P></LI></UL>
<P
align=justify>程序員使用LoadLibrary把DLL裝入內(nèi)存并且映射DLL到調(diào)用進程的虛擬地址空間(如果已經(jīng)作了映射,則增加DLL的引用計數(shù))。首先,LoadLibrary搜索DLL,搜索順序如同加載時動態(tài)鏈接一樣。然后,使用GetProcessAddress得到DLL中輸出函數(shù)的地址,并調(diào)用它。最后,使用FreeLibrary減少DLL的引用計數(shù),當(dāng)引用計數(shù)為0時,把DLL模塊從當(dāng)前進程的虛擬空間移走。</P>
<OL>
<P align=justify>
<LI>輸入庫(.lib):
<P></P>
<P align=justify>輸入庫以.lib為擴展名,格式是COFF(Common object file
format)。COFF標(biāo)準(zhǔn)庫(靜態(tài)鏈接庫)的擴展名也是.lib。COFF格式的文件可以用dumpbin來查看。</P>
<P
align=justify>輸入庫包含了DLL中的輸出函數(shù)或者輸出數(shù)據(jù)的動態(tài)鏈接信息。當(dāng)使用MFC創(chuàng)建DLL程序時,會生成輸入庫(.lib)和動態(tài)鏈接庫(.dll)。</P>
<P align=justify></P>
<LI>輸出文件(.exp)
<P></P>
<P align=justify>輸出文件以.exp為擴展名,包含了輸出的函數(shù)和數(shù)據(jù)的信息,鏈接程序使用它來創(chuàng)建DLL動態(tài)鏈接庫。</P>
<P align=justify></P>
<LI>映像文件(.map)
<P></P>
<P align=justify>映像文件以.map為擴展名,包含了如下信息:</P>
<P
align=justify>模塊名、時間戳、組列表(每一組包含了形式如section::offset的起始地址,長度、組名、類名)、公共符號列表(形式如section::offset的地址,符號名,虛擬地址flat
address,定義符號的.obj文件)、入口點如section::offset、fixup列表。</P>
<P align=justify></P>
<LI>lib.exe工具
<P></P>
<P
align=justify>它可以用來創(chuàng)建輸入庫和輸出文件。通常,不用使用lib.exe,如果工程目標(biāo)是創(chuàng)建DLL程序,鏈接程序會完成輸入庫的創(chuàng)建。</P>
<P align=justify>更詳細的信息可以參見MFC使用手冊和文檔。</P>
<P align=justify></P>
<LI>鏈接規(guī)范(Linkage Specification )
<P></P>
<P
align=justify>這是指鏈接采用不同編程語言寫的函數(shù)(Function)或者過程(Procedure)的鏈接協(xié)議。MFC所支持的鏈接規(guī)范是“C”和“C++”,缺省的是“C++”規(guī)范,如果要聲明一個“C”鏈接的函數(shù)或者變量,則一般采用如下語法:</P>
<P align=justify>#if defined(__cplusplus)</P>
<P align=justify>extern "C"</P>
<P align=justify>{</P>
<P align=justify>#endif</P>
<P align=justify></P>
<P align=justify>//函數(shù)聲明(function declarations)</P>
<P align=justify>…</P>
<P align=justify>//變量聲明(variables declarations)</P>
<P align=justify>#if defined(__cplusplus)</P>
<P align=justify>}</P>
<P align=justify>#endif</P>
<P align=justify>所有的C標(biāo)準(zhǔn)頭文件都是用如上語法聲明的,這樣它們在C++環(huán)境下可以使用。</P>
<P align=justify></P>
<LI>修飾名(Decoration name)
<P></P></LI></OL>
<P
align=justify>“C”或者“C++”函數(shù)在內(nèi)部(編譯和鏈接)通過修飾名識別。修飾名是編譯器在編譯函數(shù)定義或者原型時生成的字符串。有些情況下使用函數(shù)的修飾名是必要的,如在模塊定義文件里頭指定輸出“C++”重載函數(shù)、構(gòu)造函數(shù)、析構(gòu)函數(shù),又如在匯編代碼里調(diào)用“C””或“C++”函數(shù)等。</P>
<P align=justify>修飾名由函數(shù)名、類名、調(diào)用約定、返回類型、參數(shù)等共同決定。</P>
<OL>
<OL>
<P align=justify>
<LI><A name=_Toc445889071></A><A name=_Toc445782474></A><A
name=_Toc452640935></A><A name=_Toc457299034></A><B>調(diào)用約定</B>
<P></P></LI></OL></OL>
<P align=justify>調(diào)用約定(Calling
convention)決定以下內(nèi)容:函數(shù)參數(shù)的壓棧順序,由調(diào)用者還是被調(diào)用者把參數(shù)彈出棧,以及產(chǎn)生函數(shù)修飾名的方法。MFC支持以下調(diào)用約定:</P>
<OL>
<P align=justify>
<LI>_cdecl
<P></P>
<P
align=justify>按從右至左的順序壓參數(shù)入棧,由調(diào)用者把參數(shù)彈出棧。對于“C”函數(shù)或者變量,修飾名是在函數(shù)名前加下劃線。對于“C++”函數(shù),有所不同。</P>
<P align=justify>如函數(shù)void
test(void)的修飾名是_test;對于不屬于一個類的“C++”全局函數(shù),修飾名是?test@@ZAXXZ。</P>
<P align=justify>這是MFC缺省調(diào)用約定。由于是調(diào)用者負責(zé)把參數(shù)彈出棧,所以可以給函數(shù)定義個數(shù)不定的參數(shù),如printf函數(shù)。</P>
<P align=justify></P>
<LI>_stdcall
<P></P>
<P
align=justify>按從右至左的順序壓參數(shù)入棧,由被調(diào)用者把參數(shù)彈出棧。對于“C”函數(shù)或者變量,修飾名以下劃線為前綴,然后是函數(shù)名,然后是符號“@”及參數(shù)的字節(jié)數(shù),如函數(shù)int
func(int a, double b)的修飾名是_func@12。對于“C++”函數(shù),則有所不同。</P>
<P align=justify>所有的Win32 API函數(shù)都遵循該約定。</P>
<P align=justify></P>
<LI>_fastcall
<P></P>
<P
align=justify>頭兩個DWORD類型或者占更少字節(jié)的參數(shù)被放入ECX和EDX寄存器,其他剩下的參數(shù)按從右到左的順序壓入棧。由被調(diào)用者把參數(shù)彈出棧,對于“C”函數(shù)或者變量,修飾名以“@”為前綴,然后是函數(shù)名,接著是符號“@”及參數(shù)的字節(jié)數(shù),如函數(shù)int
func(int a, double b)的修飾名是@func@12。對于“C++”函數(shù),有所不同。</P>
<P align=justify>未來的編譯器可能使用不同的寄存器來存放參數(shù)。</P>
<P align=justify></P>
<LI>thiscall
<P></P>
<P
align=justify>僅僅應(yīng)用于“C++”成員函數(shù)。this指針存放于CX寄存器,參數(shù)從右到左壓棧。thiscall不是關(guān)鍵詞,因此不能被程序員指定。</P>
<P align=justify></P>
<LI>naked call
<P></P>
<P
align=justify>采用1-4的調(diào)用約定時,如果必要的話,進入函數(shù)時編譯器會產(chǎn)生代碼來保存ESI,EDI,EBX,EBP寄存器,退出函數(shù)時則產(chǎn)生代碼恢復(fù)這些寄存器的內(nèi)容。naked
call不產(chǎn)生這樣的代碼。</P>
<P align=justify>naked call不是類型修飾符,故必須和_declspec共同使用,如下:</P>
<P align=justify>__declspec( naked ) int func( formal_parameters )</P>
<P align=justify>{</P>
<P align=justify>// Function body</P>
<P align=justify>}</P>
<P align=justify></P>
<LI>過時的調(diào)用約定
<P></P></LI></OL>
<P align=justify>原來的一些調(diào)用約定可以不再使用。它們被定義成調(diào)用約定_stdcall或者_cdecl。例如:</P>
<P align=justify>#define CALLBACK __stdcall</P>
<P align=justify>#define WINAPI __stdcall</P>
<P align=justify>#define WINAPIV __cdecl</P>
<P align=justify>#define APIENTRY WINAPI</P>
<P align=justify>#define APIPRIVATE __stdcall</P>
<P align=justify>#define PASCAL __stdcall</P>
<P align=justify></P>
<P
align=justify>表7-1顯示了一個函數(shù)在幾種調(diào)用約定下的修飾名(表中的“C++”函數(shù)指的是“C++”全局函數(shù),不是成員函數(shù)),函數(shù)原型是void
CALLTYPE test(void),CALLTYPE可以是_cdecl、_fastcall、_stdcall。</P>
<P align=center>表7-1 不同調(diào)用約定下的修飾名</P>
<P align=center>
<CENTER>
<TABLE cellSpacing=1 cellPadding=7 width=487 border=1>
<TBODY>
<TR>
<TD vAlign=top width="24%">
<P align=justify>調(diào)用約定 </P></TD>
<TD vAlign=top width="33%">
<P align=justify>extern “C”或.C文件 </P></TD>
<TD vAlign=top width="43%">
<P align=justify>.cpp, .cxx或/TP編譯開關(guān) </P></TD></TR>
<TR>
<TD vAlign=top width="24%">
<P align=justify>_cdecl </P></TD>
<TD vAlign=top width="33%">
<P align=justify>_test </P></TD>
<TD vAlign=top width="43%">
<P align=justify>?test@@ZAXXZ </P></TD></TR>
<TR>
<TD vAlign=top width="24%">
<P align=justify>_fastcall </P></TD>
<TD vAlign=top width="33%">
<P align=justify>@test@0 </P></TD>
<TD vAlign=top width="43%">
<P align=justify>?test@@YIXXZ </P></TD></TR>
<TR>
<TD vAlign=top width="24%">
<P align=justify>_stdcall </P></TD>
<TD vAlign=top width="33%">
<P align=justify>_test@0 </P></TD>
<TD vAlign=top width="43%">
<P align=justify>?test@@YGXXZ </P></TD></TR></TBODY></TABLE></CENTER>
<P></P>
<OL>
<OL>
<OL>
<P align=justify>
<LI><A name=_Toc445889073></A><A name=_Toc445782476></A><A
name=_Toc452640937></A><A name=_Toc457299035></A><B>MFC的DLL應(yīng)用程序的類型</B>
<P></P></LI></OL></OL></OL>
<OL>
<P align=justify>
<LI>靜態(tài)鏈接到MFC的規(guī)則DLL應(yīng)用程序
<P></P>
<P align=justify>該類DLL應(yīng)用程序里頭的輸出函數(shù)可以被任意Win32程序使用,包括使用MFC的應(yīng)用程序。輸入函數(shù)有如下形式:</P>
<P align=justify>extern "C" EXPORT YourExportedFunction( );</P>
<P align=justify>如果沒有extern “C”修飾,輸出函數(shù)僅僅能從C++代碼中調(diào)用。</P>
<P align=justify>DLL應(yīng)用程序從CWinApp派生,但沒有消息循環(huán)。</P>
<P align=justify></P>
<LI>動態(tài)鏈接到MFC的規(guī)則DLL應(yīng)用程序
<P></P>
<P
align=justify>該類DLL應(yīng)用程序里頭的輸出函數(shù)可以被任意Win32程序使用,包括使用MFC的應(yīng)用程序。但是,所有從DLL輸出的函數(shù)應(yīng)該以如下語句開始:</P>
<P align=justify>AFX_MANAGE_STATE(AfxGetStaticModuleState( ))</P>
<P align=justify>此語句用來正確地切換MFC模塊狀態(tài)。關(guān)于MFC的模塊狀態(tài),后面第9章有詳細的討論。</P>
<P align=justify>其他方面同靜態(tài)鏈接到MFC的規(guī)則DLL應(yīng)用程序。</P>
<P align=justify></P>
<LI>擴展DLL應(yīng)用程序
<P></P></LI></OL>
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -