色婷婷AⅤ一区二区三区|亚洲精品第一国产综合亚AV|久久精品官方网视频|日本28视频香蕉

          新聞中心

          如何編寫(xiě)dll文件

          作者: 時(shí)間:2011-09-16 來(lái)源:網(wǎng)絡(luò) 收藏

          如何編寫(xiě):可以用幾種語(yǔ)言來(lái)實(shí)現(xiàn),如delphi編寫(xiě),pb編寫(xiě),java 編寫(xiě)dll,vc 編寫(xiě)dll,mfc dll 編寫(xiě)等。

          ——?jiǎng)討B(tài)鏈接庫(kù)(dll)是包含共享函數(shù)庫(kù)的二進(jìn)制,可以被多個(gè)應(yīng)用程序同時(shí)使用。建立應(yīng)用程序的可執(zhí)行文件時(shí),不必將DLL連接到應(yīng)用程序中,而是 在運(yùn)行時(shí)動(dòng)態(tài)裝載DLL,裝載時(shí)DLL被映射到調(diào)用進(jìn)程的地址空間中。通常我們?cè)谡{(diào)用DLL時(shí)所需的DLL文件必須位于以下三個(gè)目錄之一:
          ——(1)Windows的系統(tǒng)目錄:windowssystem;
          ——(2)DOS中path所指出的任何目錄;
          ——(3)程序所在的目錄;
          一.動(dòng)態(tài)鏈接庫(kù)(DLL)結(jié)構(gòu)
          ——DLL中定義有兩種函數(shù):導(dǎo)出函數(shù)(export function)和內(nèi)部函數(shù)
          (internal function),導(dǎo)出函數(shù)可以被其他模塊調(diào)用,內(nèi)部函數(shù)只能在DLL內(nèi)部使用。我們?cè)谟肅++定制DLL文件時(shí),需要編寫(xiě)的就是包含導(dǎo)出函數(shù)表的模塊 定義文件(.DEF)和實(shí)現(xiàn)導(dǎo)出函數(shù)功能的C++文件。下面以Sample.dll為例介紹DEF文件和實(shí)現(xiàn)文件的結(jié)構(gòu):
          ——1.模塊定義文件(.DEF)是由一個(gè)或者多個(gè)用于描述DLL屬性的模塊語(yǔ)
          句組成的文本文件,每個(gè).DEF文件至少必須包含以下模塊定義語(yǔ)句:
          第一個(gè)語(yǔ)句必須是LIBRARY語(yǔ)句,指出DLL的名字。
          EXPORTS語(yǔ)句列出被導(dǎo)出函數(shù)的名字。
          可以使用DESCRIPTION語(yǔ)句描述DLL的用途(此句可選)。
          ";"對(duì)一行進(jìn)行注釋(可選)
          ——2.實(shí)現(xiàn)文件(.cpp文件為例)
          ——實(shí)現(xiàn)入口表函數(shù)的.cpp文件中,包含DLL入口點(diǎn)處理的API函數(shù)和導(dǎo)出
          函數(shù)的代碼。
          二.創(chuàng)建Sample.dll
          ——1.首先創(chuàng)建Sample.dll的工程,啟動(dòng)VC++5.0按以下步驟生成DLL工程:
          在菜單中選擇FileNewProject
          在工程列表中選擇Win32Dynamic-LinkLibrary
          在ProjectName中輸入工程名:Sample
          單擊Location右邊按鈕,選擇c:sample目錄
          單擊OK完成,至此已創(chuàng)建了Sample.dll的工程文件
          ----2.創(chuàng)建Sample.def文件
          在菜單中選擇FileNewTextFile
          輸入以下完代碼后保存文件名"Sample.def"
          Sample.def
          指出DLL的名字Sample,鏈接器將這個(gè)名
          字放到DLL導(dǎo)入庫(kù)中
          LIBRARY Sample
          定義導(dǎo)出函數(shù)ShowMe()為例
          EXPORTS
          ShowMe
          def文件結(jié)束
          ---- 3. 創(chuàng) 建Sample.cpp
          .在菜單中選擇FileNewC++ Source File項(xiàng)
          .輸入以下代碼后保存文件名"Sample.cpp"
          //Sample.cpp
          #include < windows.h >
          int ShowMe(void);
          //DllEntryPoint為DLL入口點(diǎn)函數(shù),
          負(fù)責(zé)初試化并終止DLL
          BOOL WINAPI DllEntryPoint(HINSTANCE
          hDLL,DWORD dwReason,LPVOID Reserved)
          {
          switch(dwReason)
          {
          case DLL_PROCESS_ATTACH:
          {
          break;
          }
          case DLL_PROCESS_DETACH:
          {
          break;
          }
          }
          return TRUE;
          }
          int ShowMe(void)
          {
          //蜂鳴器響一下
          MessageBeep((WORD)-1);
          MessageBox("你好!");
          return 1;
          }
          ——4.編譯DLL文件——從 Build 菜單中選擇 BuildSample.DLL, 產(chǎn)生
          Sample.DLL文件,以后就可以隨時(shí)調(diào)用了。
          三.在應(yīng)用程序中調(diào)用DLL文件
          ——在應(yīng)用程序中要首先裝入DLL后才能調(diào)用導(dǎo)出表中的函數(shù),例如用MFC
          創(chuàng)建基于對(duì)話框的工程Test,并在對(duì)話框上放置"Load"按鈕,你就必須添加裝載代碼。----1.首先在TestDlg.cpp的首部添加變量設(shè)置代碼:
          //設(shè)置全局變量gLibSample用于存儲(chǔ)DLL句柄
          HINSTANCE gLibSample=NULL;
          //第二個(gè)變量ShowMe是指向DLL
          庫(kù)中ShowMe()函數(shù)的指針
          typedef int(* SHOWME)(void);
          SHOWME ShowMe;
          2.利用ClassWizard為"Load"按鈕添加裝載DLL的代碼
          Void CTestDlg::OnLoadButton()
          {
          //要添加的代碼如下
          if(gLibMyDLL!=NULL)
          {
          MessageBox("The Sample.DLL has already been load.");
          return;
          }
          //裝載Sample.dll,未加路徑,將在三個(gè)默認(rèn)路徑中尋找
          gLibSample=LoadLibrary("SAMPLE.DLL");
          //返回DLL中ShowMe()函數(shù)的地址
          ShowMe=(SHOWME)
          GetProcAddress(gLibSample,"ShowMe");
          //代碼添加完畢
          }
          ——3.只要DLL裝載成功,在應(yīng)用程序中就可以直接調(diào)用ShowMe()函數(shù),此時(shí)已完成了定制和調(diào)用DLL的全部過(guò)程。----本程序在Windows95,VC++5.0中運(yùn)行通過(guò)。
          生成一個(gè)Dll工程,然后將你的類和函數(shù)都添加進(jìn)去,然后將要輸出的函數(shù)在該工程下的Translate.def文件中將名字列出(注意,只要名字,不要括號(hào)、參數(shù)等);實(shí)際上和你的Exe工程沒(méi)有很大的區(qū)別。
          如果你要用VB用,在Dll中定義你的函數(shù)的時(shí)候前面要加上“__stdcall”關(guān)鍵字,否則VB沒(méi)有辦法使用。
          建議Dll中少用資源,只用函數(shù)比較好,如果要用資源,在資源載入時(shí)要做一個(gè)處理,不然會(huì)由于hModule的問(wèn)題發(fā)生錯(cuò)誤。
          第十三章 動(dòng)態(tài)鏈接庫(kù)
          在Windows應(yīng)用程序中使用動(dòng)態(tài)鏈接庫(kù)有很多的好處。最主要的一點(diǎn)說(shuō)是它可以使得多個(gè)應(yīng)用程序共享一段代碼,從而可以大幅度的降低應(yīng)用程序的資源開(kāi) 銷,同時(shí)很縮小了應(yīng)用程序的最終執(zhí)行代碼的大小。此外,通過(guò)使用動(dòng)態(tài)鏈接庫(kù),我們可以把一些常規(guī)的例程獨(dú)立出來(lái),有效的避免了不必要的重復(fù)開(kāi)發(fā),并且,由 于應(yīng)用程序使用了動(dòng)態(tài)鏈接的方式,還可以在不需重新改寫(xiě)甚至編譯應(yīng)用程序的基礎(chǔ)上更新應(yīng)用程序的某些組件。
          本章介紹Visual C++ 5.0中的動(dòng)態(tài)鏈接庫(kù)的創(chuàng)建和使用。這些內(nèi)容包括
          為什么要使用DLL
          不同的DLL類型之間的選擇
          在程序中創(chuàng)建和使用DLL
          使用DLL擴(kuò)展MFC
          本章的重點(diǎn)在講述Win32 DLL和基于MFC的常規(guī)DLL,對(duì)于MFC擴(kuò)展DLL僅給了一個(gè)很簡(jiǎn)單的例子,更深入的資料請(qǐng)參閱Visual C++的聯(lián)機(jī)文檔。
          第一節(jié) 概述
          動(dòng)態(tài)鏈接庫(kù)(DLL,Dynamic-Link Library)也是一種可執(zhí)行文件,只不過(guò)它不能象普通的EXE文件那樣可以直接運(yùn)行,而是用來(lái)為其它可執(zhí)行文件(包括EXE文件和其它DLL)提供共 享函數(shù)庫(kù)。使用DLL的應(yīng)用程序可以調(diào)用DLL中的導(dǎo)出函數(shù)(imported function),不過(guò)在應(yīng)用程序本身的執(zhí)行代碼中并不包含這些函數(shù)的執(zhí)行代碼,它們經(jīng)過(guò)編譯和鏈接之后,獨(dú)立的保存在DLL中。這是一種和過(guò)去常用的 靜態(tài)鏈接不同的方式,使用靜態(tài)鏈接庫(kù)(static link library)的應(yīng)用程序從函數(shù)庫(kù)得到所引用的函數(shù)的執(zhí)行代碼,然后把執(zhí)行代碼放進(jìn)自身的執(zhí)行文件中,這樣,應(yīng)用程序在運(yùn)行時(shí)就可以不再需要靜態(tài)函數(shù)庫(kù) 的支持。而使用DLL的應(yīng)用程序只包括了用于從DLL中定位所引用的函數(shù)的信息,而沒(méi)有函數(shù)具體實(shí)現(xiàn),要等到程序運(yùn)行時(shí)才從DLL中獲得函數(shù)的實(shí)現(xiàn)代碼。 顯然,使用了DLL的應(yīng)用程序在運(yùn)行時(shí)必須要有相應(yīng)的DLL的支持。
          DLL在Windows編程中得到了廣泛的應(yīng)用。Windows應(yīng)用 程序的基礎(chǔ):Windows API函數(shù)中的相當(dāng)部分就是由一組DLL所提供的,這些DLL從安裝Windows起就存在于系統(tǒng)中了。盡管在前面的幾章中我們沒(méi)有明確的提到,但事實(shí)上 我們?cè)缇驮谑褂肈LL進(jìn)行編程了,只不過(guò),所使用的DLL都是現(xiàn)成的,并且所有調(diào)用DLL的操作都由Visual C++的編譯和鏈接程序替我們完成了。
          本章將詳細(xì)的介紹如何創(chuàng)建自己的DLL和如何使用自己創(chuàng)建的DLL進(jìn)行編程。
          為什么要使用DLL呢?這當(dāng)然是因?yàn)榕c傳統(tǒng)的靜態(tài)鏈庫(kù)相比,使用DLL有著更多的優(yōu)勢(shì):
          使用DLL提供了一種共享數(shù)據(jù)和代碼的方便途徑。并且,由于多個(gè)應(yīng)用程序可以共享同一個(gè)DLL中的函數(shù),因此,使用DLL可以顯著的節(jié)省磁盤(pán)空間。尤其 對(duì)于Windows應(yīng)用程序,有很多的操作都是“標(biāo)準(zhǔn)化”了的,如果使用傳統(tǒng)的靜態(tài)鏈接,每一個(gè)需要完成這些操作的應(yīng)用程序都必須在自己的執(zhí)行文件中包括 相同的執(zhí)行代碼,這不但使單個(gè)的應(yīng)用程序變得更長(zhǎng),也浪費(fèi)了磁盤(pán)空間。
          由于相同的原因,多個(gè)應(yīng)用程序還可以同時(shí)共享DLL在內(nèi)存中的同一份拷貝,這樣就有效的節(jié)省了應(yīng)用程序所占用的內(nèi)存資源,減少了頻繁的內(nèi)存交換,從而提高了應(yīng)用程序的執(zhí)行效率。
          由于DLL是獨(dú)立于可執(zhí)行文件的,因此,如果需要向DLL中增加新的函數(shù)或是增強(qiáng)現(xiàn)有函數(shù)的功能,只要原有函數(shù)的參數(shù)和返回值等屬性不變,那么,所有使 用該DLL的原有應(yīng)用程序都可以在升級(jí)后的DLL的支持下運(yùn)行,而不需要重新編譯。這就極大的方便了應(yīng)用程序的升級(jí)和售后支持。
          DLL除了包括函數(shù)的執(zhí)行代碼以外,還可以只包括如圖標(biāo)、位圖、字符串和對(duì)話框之類的資源,因此可以把應(yīng)用程序所使用的資源獨(dú)立出來(lái)做成DLL。對(duì)于一些常用的資源,把它們做到DLL中后,就可為多個(gè)應(yīng)用程序所共享。
          便于建立多語(yǔ)言的應(yīng)用程序。我們可以把多語(yǔ)言應(yīng)用程序中所使用的與語(yǔ)言相關(guān)的函數(shù)做到DLL中,只要不同語(yǔ)言的應(yīng)用程序所調(diào)用的函數(shù)都具有相同的接口,這樣就可以通過(guò)簡(jiǎn)單地更換DLL來(lái)實(shí)現(xiàn)多語(yǔ)言支持。
          然而,使用DLL也有其不足之處。最典型的就是應(yīng)用程序在運(yùn)行時(shí)必須要有相應(yīng)的DLL的支持。另外,使用DLL也增大了程序運(yùn)行的開(kāi)銷,好在這種額外的開(kāi)銷對(duì)于大多數(shù)應(yīng)用程序的影響并不明顯,我們也只是在某些對(duì)運(yùn)行速度要求苛刻的特殊場(chǎng)合,才不得不考慮這一點(diǎn)。
          Visual C++ 5.0支持多種DLL,包括:
          非MFC DLL
          靜態(tài)鏈接到MFC的常規(guī)DLL
          動(dòng)態(tài)鏈接到MFC的常規(guī)DLL
          MFC擴(kuò)展DLL
          其中非MFC DLL(non-MFC DLL)內(nèi)部不使用MFC,調(diào)用非MFC DLL提供的導(dǎo)出函數(shù)的可執(zhí)行程序可以使用MFC,也可以不使用MFC。一般來(lái)說(shuō),非MFC DLL的導(dǎo)出函數(shù)都使用標(biāo)準(zhǔn)的C接口(standard C interface)。
          其余三種DLL的內(nèi)部都使用了MFC。顧名思義,靜態(tài)鏈接到MFC的常規(guī)DLL(regular DLL statically linking to MFC)和動(dòng)態(tài)鏈接到MFC的常規(guī)DLL(regular DLL dynamically linking to MFC)的區(qū)別在于一個(gè)使用的是MFC的靜態(tài)鏈接庫(kù),而另一個(gè)使用的是MFC的DLL。這和一般的MFC應(yīng)用程序的情況是很類似的。
          MFC擴(kuò)展DLL一般用來(lái)提供派生于MFC的可重用的類,以擴(kuò)展已有的MFC類庫(kù)的功能。MFC擴(kuò)展DLL使用MFC的動(dòng)態(tài)鏈接版本。只有使用MFC動(dòng)態(tài) 鏈接的可執(zhí)行程序(無(wú)論是EXE還是DLL)才能訪問(wèn)MFC擴(kuò)展DLL。MFC擴(kuò)展DLL的另一個(gè)有用的功能是它可以在應(yīng)用程序和它所加載的MFC擴(kuò)展 DLL之間傳遞MFC和MFC派生對(duì)象的指針。在其它情況下,這樣做是可能導(dǎo)致問(wèn)題的。
          注意:
          只有Visual C++ 5.0的專業(yè)版和企業(yè)版才支持到MFC的靜態(tài)鏈接。
          靜態(tài)鏈接到MFC的常規(guī)DLL過(guò)去的USRDLL有著相同的特性,同樣的,MFC擴(kuò)展DLL和過(guò)去的AFXDLL有著相同的特性。在Visual C++ 5.0中已不再使用USRDLL和AFXDLL這兩個(gè)術(shù)語(yǔ)。
          如何選擇所應(yīng)該使用的DLL的類型呢?我們可以從以下幾個(gè)方面來(lái)考慮:
          相比使用了MFC的DLL而言,非MFC DLL顯得更為短小精悍。因此,如果DLL不需要使用MFC,那么使用非MFC DLL是一個(gè)很好的選擇,它將顯著地節(jié)省磁盤(pán)和內(nèi)存空間。同時(shí),無(wú)論應(yīng)用程序是否使用了MFC,都可以調(diào)用非MFC DLL中所導(dǎo)出的函數(shù)。
          如果需要?jiǎng)?chuàng)建使用了MFC的DLL,并希望MFC和非MFC應(yīng)用程序都能使用所創(chuàng)建的DLL,那么可以選擇的范圍包括靜態(tài)鏈接到MFC的常規(guī)DLL和動(dòng) 態(tài)鏈接到MFC的常規(guī)DLL。動(dòng)態(tài)鏈接到MFC的常規(guī)DLL比較短小,因此可以節(jié)省磁盤(pán)和內(nèi)存,但是,在分發(fā)動(dòng)態(tài)鏈接到MFC的常規(guī)DLL時(shí),必須同時(shí)分 發(fā)MFC的支持DLL,如MFCx0.DLL和MSVCRT.DLL等。而使用靜態(tài)鏈接到MFC的常規(guī)DLL則不存在這種問(wèn)題。
          如果希望在DLL中實(shí)現(xiàn)從MFC派生的可重用的類,或者是希望在應(yīng)用程序和DLL之間傳遞MFC的派生對(duì)象時(shí),必須選擇MFC擴(kuò)展DLL。
          第二節(jié) 創(chuàng)建和使用動(dòng)態(tài)鏈接庫(kù)
          本節(jié)以非MFC DLL為例來(lái)講解DLL的結(jié)構(gòu)和導(dǎo)出方法,并介紹創(chuàng)建和使用DLL的方法和步驟。
          13.2.1 DLL的結(jié)構(gòu)和導(dǎo)出方式
          DLL文件和EXE文件都屬于可執(zhí)行文件,不同的是DLL文件包括了一個(gè)導(dǎo)出表,導(dǎo)出表中給出了可以從DLL中導(dǎo)出的所有函數(shù)的名字。外部可執(zhí)行程序只能訪問(wèn)包括在DLL的導(dǎo)出表中的函數(shù),DLL中的其它函數(shù)是私有的,不能為外部可執(zhí)行程序所訪問(wèn)。
          可以使用Visual C++提供的DUMPBIN實(shí)用程序(可以在DevStudioVCbin目錄下找到這個(gè)工具)來(lái)查看一個(gè)DLL文件的結(jié)構(gòu)。舉一個(gè)例子,如果需要查 看DLL文件msgbox.dll(我們將在本小節(jié)的后續(xù)內(nèi)容中創(chuàng)建該DLL)的導(dǎo)出表,可以在命令提示符下鍵入下面的命令:
          >dumpbin /exports msgbox.dll
          運(yùn)行結(jié)果如下:
          Microsoft (R) COFF Binary File Dumper Version 5.00.7022
          Copyright (C) Microsoft Corp 1992-1997. All rights reserved.
          Dump of file msgbox.dll
          File Type: DLL
          Section contains the following Exports for MSGBOX.dll
          0 characteristics
          351643C3 time date stamp Mon Mar 23 19:13:07 1998
          0.00 version
          1 ordinal base
          1 number of functions
          1 number of names
          ordinal hint name
          1 0 MsgBox (00001000)
          Summary
          7000 .data
          1000 .idata
          2000 .rdata
          2000 .reloc
          17000 .text
          由上面的結(jié)果得知,msgbox.dll中僅包括了一個(gè)導(dǎo)出函數(shù)MsgBox()。
          注意:
          僅僅知道導(dǎo)出函數(shù)的名稱并不足以從DLL中導(dǎo)出該函數(shù)。若在應(yīng)用程序中使用顯式鏈接(link explicitly),至少還應(yīng)該知道導(dǎo)出函數(shù)的返回值的類型以及所傳遞給導(dǎo)出函數(shù)的參數(shù)的個(gè)數(shù)、順序和類型;若使用隱含鏈接(link implicitly),必須有包括導(dǎo)出函數(shù)(或類)的定義的頭文件(.H文件)和引入庫(kù)(import library,.LIB文件),這些文件是由DLL的創(chuàng)建者所提供的。關(guān)于顯式鏈接和隱含鏈接,將在本章的“13.2.2 鏈接應(yīng)用程序到DLL”小節(jié)中講述。
          從DLL中導(dǎo)出函數(shù)有兩種方法:
          在創(chuàng)建DLL時(shí)使用模塊定義(module DEFinition,.DEF)文件。
          在定義函數(shù)時(shí)使用關(guān)鍵字__declspec(dllexport)。
          下面我們通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)分別說(shuō)明兩種方法的使用。在這個(gè)例子中,我們將創(chuàng)建一個(gè)只包括一個(gè)函數(shù)MsgBox()的DLL,函數(shù)MsgBox()用 來(lái)顯示一個(gè)消息框,它和Win32 API函數(shù)MessageBox()的功能是一樣,只不過(guò)在函數(shù)MsgBox()中,不需要指定消息的父窗口,而且可以缺省其它所有的參數(shù)。
          (1) 使用模塊定義文件
          模塊定義文件是一個(gè)文本文件,它包括了一系列的模塊語(yǔ)句,這些語(yǔ)句用來(lái)描述DLL的各種屬性,典型的,模塊語(yǔ)句定義了DLL中所導(dǎo)出的函數(shù)的名稱和順序值。
          在講解模塊定義文件之前,我們先創(chuàng)建一個(gè)Win32 Dynamic-Link Library工程。
          1. 在Microsoft Developer Studio中選擇File菜單下的New命令,在Projects選項(xiàng)卡中選擇Win32 Dynamic-Link Library,并為工程取一個(gè)名字,如msgbox。單擊OK后,Visual C++創(chuàng)建一個(gè)Win32 DLL的空白工程,必須手動(dòng)的將所需要的文件添加到工程中。
          2. 單擊Project菜單下的Add To Project子菜單下的New命令,在Files選項(xiàng)卡中選擇Text File,在File文本框中輸入DEF文件名,如msgbox.def。
          3. 雙擊Workspace窗口的FileView選項(xiàng)卡中的msgbox.def節(jié)點(diǎn),在msgbox.def文件中輸入下面的內(nèi)容:
          LIBRARY MSGBOX
          DESCRIPTION "一個(gè)DLL的簡(jiǎn)單例子"
          EXPORTS
          MsgBox @1
          在DEF文件中的第一條語(yǔ)句必須是LIBRARY語(yǔ)句,該語(yǔ)句表明該DEF文件屬于一個(gè)DLL,在LIBRARY之后是DLL的名稱,這個(gè)名稱在鏈接時(shí)將放到DLL的引入庫(kù)中。
          EXPORTS語(yǔ)句下列出了DLL的所有導(dǎo)出函數(shù)以及它們的順序值。函數(shù)的順序值不是必須的,在指定導(dǎo)出函數(shù)的順序值時(shí),我們?cè)诤瘮?shù)名后跟上一個(gè)@符號(hào) 和一個(gè)數(shù)字,該數(shù)字即導(dǎo)出函數(shù)的順序值。如果在DEF中指定了順序值,它必須不小于1,且不大于DLL中所有導(dǎo)出函數(shù)的數(shù)目。
          DESCRIPTION語(yǔ)句是可選的,它簡(jiǎn)單的說(shuō)明了DLL的用途。
          4. 下一步是向工程中添加一個(gè)頭文件,它定義了DLL中的函數(shù)的返回值的類型和參數(shù)的個(gè)數(shù)、順序和類型。
          單擊菜單項(xiàng)Project|Add To Project|New...,在Files選項(xiàng)卡下選擇C/C++ Header File,在File文本框中指定頭文件名,如msgbox.h(可以省略后綴名.h)。
          在頭文件中輸入如下的內(nèi)容:
          #include
          extern "C" int MsgBox(
          // 消息框的文本
          LPCTSTR lpText="雖然這個(gè)例子有一些幼稚,但它工作得非常的好!",
          // 消息框的標(biāo)題
          LPCTSTR lpCaption="一個(gè)簡(jiǎn)單的例子",
          // 消息框的樣式
          UINT uType=MB_OK);
          請(qǐng)注意函數(shù)定義前的關(guān)鍵字extern "C",這是由于我們使用了C++語(yǔ)言來(lái)開(kāi)發(fā)DLL,為了使C語(yǔ)言模塊能夠訪問(wèn)該導(dǎo)出函數(shù),我們應(yīng)該使用C鏈接來(lái)代替C++鏈接。否則,C++編譯器將使 用C++的類型安全命名和調(diào)用協(xié)議,這在使用C調(diào)用該函數(shù)時(shí)就會(huì)遇上問(wèn)題。在本例中并不需要考慮到這個(gè)問(wèn)題,因?yàn)槲覀冊(cè)陂_(kāi)發(fā)DLL和應(yīng)用程序時(shí)都是使用C ++,但我們?nèi)匀粡?qiáng)烈建議使用extern "C",以保證在使用C編寫(xiě)的程序調(diào)用該DLL的導(dǎo)出函數(shù)不會(huì)遇上麻煩,在本章后面的內(nèi)容中我們還會(huì)討論到這個(gè)問(wèn)題。
          5. 下面要做的事是向工程中添加一個(gè)C++源文件,在該文件中實(shí)現(xiàn)函數(shù)MsgBox()。
          仿照上面的過(guò)程,單擊菜單項(xiàng)Project|Add To Project|New...,在Files選項(xiàng)卡下選擇C++ Source File,在File文本框中指定源文件名,如msgbox.cpp。
          在msgbox.cpp文件中添加如下的代碼:
          #include "test1.h"
          int MsgBox(LPCTSTR lpText,
          LPCTSTR lpCaption,
          UINT uType)
          {
          return MessageBox(NULL,lpText,lpCaption,uType);
          }
          編譯該工程,在Debug目錄下生成文件msgbox.lib和msgbox.dll。
          在“13.2.2 鏈接應(yīng)用程序到DLL”小節(jié)中將講述如何使用在本節(jié)中所創(chuàng)建的DLL:msgbox.dll。
          (2) 使用關(guān)鍵字__declspec(dllexport)
          從DLL中導(dǎo)出文件的另一種方法是在定義函數(shù)時(shí)使用__declspec(dllexport)關(guān)鍵字。這種方法不需要使用DEF文件。
          仍使用前面的例子,在工程中刪除msgbox.def文件,將msgbox.h文件修改如下:
          #include
          extern "C" __declspec(dllexport) int MsgBox(
          // 消息框的文本
          LPCTSTR lpText="雖然這個(gè)例子有一些幼稚,但它工作得非常的好!",
          // 消息框的標(biāo)題
          LPCTSTR lpCaption="一個(gè)簡(jiǎn)單的例子",
          // 消息框的樣式
          UINT uType=MB_OK);
          msgbox.cpp文件并不需要做任何修改,重新編譯該工程,在Debug目錄下仍生成兩個(gè)文件msgbox.lib和msgbox.dll。
          在下一小節(jié)“13.2.2 鏈接應(yīng)用程序到DLL”中講述了如何在應(yīng)用程序中使用所創(chuàng)建的DLL:msgbox.dll的導(dǎo)出函數(shù)MsgBox()。
          使用__declspec(dllexport)從DLL中導(dǎo)出類的語(yǔ)法如下:
          class __declspec(dllexport) CDemoClass
          {
          ...
          }
          注意:
          如果在使用__declspec(dllexport)的同時(shí)指定了調(diào)用協(xié)議關(guān)鍵字,則必須將__declspec(dllexport)關(guān)鍵字放在調(diào)用協(xié)議關(guān)鍵字的左邊。如:
          int __declspec(dllexport) __cdacl MyFunc();
          在32位版本的Visual C++中,__declspec(dllexport)和__declspec(dllimport)代替了16版本中使用的__export關(guān)鍵字。因 此,在將16位的DLL源代碼移植到Win32平臺(tái)時(shí),需要把每一處__export替換為_(kāi)_declsped(dllexport)。
          如何從這兩種導(dǎo)出函數(shù)的方法中作出選擇,可以從下面的幾個(gè)方面考慮:
          如果需要使用導(dǎo)出順序值(export ordinal value),那么應(yīng)該使用DEF文件來(lái)導(dǎo)出函數(shù)。只在使用DEF文件導(dǎo)出函數(shù)才能指定導(dǎo)出函數(shù)的順序值。使用順序值的一個(gè)好處是當(dāng)向DLL中添加新的函 數(shù)時(shí),只要新的導(dǎo)出函數(shù)的順序值大于原有的導(dǎo)出函數(shù),就沒(méi)有必要重新鏈接使用隱含鏈接的應(yīng)用程序。相反,如果使用__declspec (dllexport)來(lái)導(dǎo)出函數(shù),如果向DLL中添加了新的函數(shù),使用隱含鏈接的應(yīng)用程序有可以需要重新編譯和鏈接。
          使用DEF文件來(lái)導(dǎo)出函數(shù),可以創(chuàng)建具有NONAME屬性的DLL。具有NONAME屬性的DLL在導(dǎo)出表中僅包含了導(dǎo)出函數(shù)的順序值,這種類型的DLL在包括有大量的導(dǎo)出函數(shù)時(shí),其文件長(zhǎng)度要小于通常的DLL。
          使用DEF文件從C++文件導(dǎo)出函數(shù),應(yīng)該在定義函數(shù)時(shí)使用extern "C"或者在DEF文件中指定導(dǎo)出函數(shù)的decorated name。否則,由于編譯器所產(chǎn)生的decorated name是基于特定編譯器的,鏈接到該DLL的應(yīng)用程序也必須使用創(chuàng)建DLL的同一版本的Visual C++來(lái)編譯和鏈接。
          由于使用__declspec(dllexport)關(guān)鍵字導(dǎo)出函數(shù)不需要編寫(xiě)DEF文件,因此,如果編寫(xiě)的DLL只供自己使用,使用__declspec(dllexport)較為簡(jiǎn)單。

          注意:
          MFC本身使用了DEF文件從MFCx0.DLL中導(dǎo)出函數(shù)和類。
          13.2.2 鏈接應(yīng)用程序到DLL
          同樣,鏈接應(yīng)用程序到DLL也有兩種方法:
          隱含鏈接
          顯式鏈接
          隱含鏈接有時(shí)又稱為靜態(tài)加載。如果應(yīng)用程序使用了隱含鏈接,操作系統(tǒng)在加載應(yīng)用程序的同時(shí)加載應(yīng)用程序所使用的DLL。顯式鏈接有時(shí)又稱為動(dòng)態(tài)加載。使 用動(dòng)態(tài)加載的應(yīng)用程序必須在代碼中明確的加載所使用的DLL,并使用指針來(lái)調(diào)用DLL中的導(dǎo)出函數(shù),在使用完畢之后,應(yīng)用程序必須卸載所使用的DLL。
          同一個(gè)DLL可以被應(yīng)用程序隱含鏈接,也可以被顯式鏈接,這取決于應(yīng)用程序的目的和實(shí)現(xiàn)。
          下面我們?cè)诜謩e講述兩種不同的鏈接方式之后再作對(duì)比。
          (1) 使用隱含鏈接
          在使用隱含鏈接除了需要相應(yīng)的DLL文件外,還必須具備如下的條件:
          一個(gè)包括導(dǎo)出的函數(shù)或C++類的頭文件
          一個(gè)輸入庫(kù)文件(.LIB文件)
          通常情況下,我們需要從DLL的提供者那里得到上面的文件。輸入庫(kù)文件是在DLL文件被鏈接時(shí)由鏈接程序生成的。
          在“13.2.1 DLL的結(jié)構(gòu)和導(dǎo)出方式”中所創(chuàng)建的DLL:msgbox.dll所對(duì)應(yīng)的頭文件msgbox.h如下:
          #include
          extern "C" __declspec(dllimport) int MsgBox(
          // 消息框的文本
          LPCTSTR lpText="雖然這個(gè)例子有一些幼稚,但它工作得非常的好!",
          // 消息框的標(biāo)題
          LPCTSTR lpCaption="一個(gè)簡(jiǎn)單的例子",
          // 消息框的樣式
          UINT uType=MB_OK);
          需要注意的是,這個(gè)msgbox.h文件和創(chuàng)建DLL時(shí)所使用msgbox.h是不同的,唯一的差別在于,創(chuàng)建DLL時(shí)的msgbox.h中使用的是 __declspec(dllexport)關(guān)鍵字,而供應(yīng)用程序所使用的msgbox.h中使用的是__declspec(dllimport)關(guān)鍵 字。無(wú)論創(chuàng)建DLL時(shí)使用的是DEF文件還是__declspec(dllexport)關(guān)鍵字,均可使用__declspec(dllimport)關(guān) 鍵字從DLL中引入函數(shù)。引入函數(shù)時(shí)也可以省略__declspec(dllimport)關(guān)鍵字,但是使用它可以使編譯器生成效率更高的代碼。
          注意:
          如果需要引入的是DLL中的公用數(shù)據(jù)和對(duì)象,則必須使用__declspec(dllimport)關(guān)鍵字。
          現(xiàn)在使用Microsoft Developer Studio創(chuàng)建一個(gè)Win32 Application工程,命名為tester。向工程中添加一個(gè)C++源文件,如tester.cpp。在tester.cpp文件中輸入下面的代碼:
          #include "msgbox.h" // 應(yīng)將msgbox.h文件拷貝到工程tester的目錄下。
          int WINAPI WinMain(HINSTANCE hInstance,
          HINSTANCE hPrevInstance,
          LPSTR lpCmdLine,
          int nCmdShow)
          {
          return MsgBox();
          }
          在上面的代碼中,MsgBox()函數(shù)的所有參數(shù)都使用了缺省值。

          c++相關(guān)文章:c++教程


          蜂鳴器相關(guān)文章:蜂鳴器原理


          關(guān)鍵詞: dll 文件

          評(píng)論


          相關(guān)推薦

          技術(shù)專區(qū)

          關(guān)閉