• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            無我

            讓內心永遠燃燒著偉大的光明的精神之火!
            靈活的思考,嚴謹的實現
            豪邁的氣魄、頑強的意志和周全的思考

            dll的概念、dll導出類(轉)

            dll的概念、dll導出類(轉)

            Posted on 2011-07-22 14:59 白耘 閱讀(875) 評論(0) 編輯 收藏 來源:http://www.cnblogs.com/whiteyun/archive/2011/07/22/2113668.html

            1、 DLL的概念

            DLL(Dynamic Linkable Library),動態鏈接庫,可以向程序提供一些函數、變量或類。這些可以直接拿來使用。

            靜態鏈接庫與動態鏈接庫的區別:

             

            (1)靜態鏈接庫與動態鏈接庫都是共享代碼的方式。靜態鏈接庫把最后的指令都包含在最終生成的EXE文件中了;動態鏈接庫不必被包含在最終EXE文件中,EXE文件執行時可以“動態”地引用和卸載這個與EXE獨立的DLL文件。

             

            (2)靜態鏈接庫中不能再包含其他的動態鏈接庫或者靜態庫,而在動態鏈接庫中還可以再包含其他的動態或靜態鏈接庫。

             

            動態鏈接庫的分類:Visual C++支持三種DLL,它們分別是:

            (1)Non-MFC DLL(非MFC動態庫)

            非MFC動態庫不采用MFC類庫結構,其導出函數為標準的C接口,能被非MFC或MFC編寫的應用程序所調用;

             

            (2)MFC Regular DLL(MFC規則DLL)

            MFC規則DLL 包含一個繼承自CWinApp的類,但其無消息循環;

             

            (3)MFC Extension DLL(MFC擴展DLL):MFC擴展DLL采用MFC的動態鏈接版本創建,它只能被用MFC類庫所編寫的應用程序所調用。

             

            2、 創建一個DLL

             

            2.1 非MFC的DLL

             

            2.1.1聲明導出函數:

             

            extern “C” __declspec(dllexport) int add(int a, int b);

             

            其中 extern “C”為聲明為C編譯。由于C++編譯器在編譯的時候會造成其函數名的該變,在其他應用程序中導致函數不可調用,而C編譯器則不會在編譯后改變其函數名。這樣如果用C編譯的程序來調用該dll中的函數時,可能會造成找不到該函數。

             

            __declspec(dllexport)表示該函數為DLL輸出函數,即其他應用程序可以調用該函數

             

            從dll中聲明輸出函數有兩種方式:
            (1) 用模塊定義(.def) 文件聲明,.def文件為鏈接器提供了有關被鏈接程序的導出、屬性及其他方面的信息。
            (2) 用__declspec(dllexport)來聲明函數

             

            如果使用Visual C++來創建dll,對于同樣用VC創建的exe來說,調用dll沒有什么問題。

            而如果用其他工具來創建的exe來調用dll,就會出現問題。因為即使你不用C++編譯器,Microsoft C編譯器也會損害C函數。

            當用__stdcall將函數輸出時,C編譯器會將函數改為_func@1的形式。在這里需要在.def文件中加入EXPORTS節來輸出函數:

             

            EXPORTS
            func
            這樣,dll將用func函數名來輸出函數。

             

            另一種方式是用#pragma (linker, “/exports:func=_func@1”),告訴編譯器輸出函數func,這種方式沒有前一種好。

             

            如果通過VC++編寫的DLL欲被其他語言編寫的程序調用,應將函數的調用方式聲明為__stdcall方式。

            WINAPI都采用這種方式,而C/C++ 缺省的調用方式卻為__cdecl。__stdcall方式與__cdecl對函數名最終生成符號的方式不同。

            若采用C編譯方式(在C++中需將函數聲明為extern "C"),__stdcall調用約定在輸出函數名前面加下劃線,后面加“@”符號和參數的字節數,

            形如_functionname@number;而__cdecl調用約定僅在輸出函數名前面加下劃線,形如_functionname。

             

            lib.def : 導出DLL函數
            LIBRARY dllTest
            EXPORTS
            add @ 1
            .def文件的規則為:
               (1)LIBRARY語句說明.def文件相應的DLL;
               (2)EXPORTS語句后列出要導出函數的名稱。可以在.def文件中的導出函數名后加@n,表示要導出函數的序號為n

            (在進行函數調用時,這個序號將發揮其作用);

               (3).def 文件中的注釋由每個注釋行開始處的分號 (;) 指定,且注釋不能與語句共享一行。
               由此可以看出,例子中lib.def文件的含義為生成名為“dllTest”的動態鏈接庫,導出其中的add函數,并指定add函數的序號為1。

             

            2.1.2 Dll的調用方式: 

             

            DLL的調用分為兩種方式:動態和靜態顯示動態鏈接隱式動態鏈接)

             

            (1) 動態調用(顯示動態鏈接):

            typedef int(*lpAddFun)(int, int); //宏定義函數指針類型

            lpAddFun add;//函數指針
            HINSTANCE hDll=LoadLibrary(“path”);
            add=(lpAddFun)GetProcAddress(hDll, "add");//獲得dll中的add函數指針
            FreeLibrary(hDll);

             

            在從dll調用中返回的函數、指針或者類都是以指針的方式的,即返回的是函數、變量或類的地址。這里一定要注意,不能簡單的用函數名來賦值。

             

            (2) 靜態調用(隱式動態鏈接):

            將生成的.dll和.lib文件拷入到調用dll的工程中,用命令

            #pragma comment(lib,"dllTest.lib"),實現靜態調用,即把該dll在編譯的時候也編譯到exe文件中去,而后在工程中調用時用下面的代碼:

             

            #pragma comment(lib,"dllTest.lib")//.lib文件中僅僅是關于其對應DLL文件中函數的重定位信息
            extern "C" __declspec(dllimport) add(int x,int y);
            int main(int argc, char* argv[])
            {
            int result = add(2,3);
            printf("%d",result);
            return 0;
            }

             

               由上述代碼可以看出,靜態調用方式的順利進行需要完成兩個動作:
               (1) 告訴編譯器與DLL相對應的.lib文件所在的路徑及文件名,#pragma comment(lib,"dllTest.lib")就是起這個作用。程序員在 建立一個DLL文件時,連接器會自動為其生成一個對應的.lib文件,該文件包含了DLL 導出函數的符號名及序號(并不含有實際的代碼)。在應用程序 里,.lib文件將作為DLL的替代文件參與編譯。另外一種顯式調用的方式是設置vc中的目錄和includefiles來實現

             

               (2) 聲明導入函數,extern "C" __declspec(dllimport) add(int x,int y)語句中的__declspec(dllimport)發揮這個作 用。靜態調用方式不再需要使用系統API來加載、卸載DLL以及獲取DLL中導出函數的地址。這是因為,當程序員通過靜態鏈接方式編譯生成應用程序時,應 用程序中調用的與.lib文件中導出符號相匹配的函數符號將進入到生成的EXE 文件中,.lib文件中所包含的與之對應的DLL文件的文件名也被編譯器存儲在 EXE文件內部。當應用程序運行過程中需要加載DLL文件時,Windows將根據這些信息發現并加載DLL,然后通過符號名實現對DLL 函數的動態鏈接。這 樣,EXE將能直接通過函數名調用DLL的輸出函數,就象調用程序內部的其他函數一樣。

             

            2.1.3 DllMain函數

             

            Windows在加載dll的時候,會首先需要一個入口函數DllMain。當在dll中不定義DllMain的時候,windows會從其他運行庫中調用一個不做任何操作的DllMain函數,直接返回true。DllMain是dll內部的函數,這意味著在調用dll的程序中不能顯式的調用。它是在 dll被調用時自動被調用的。
             

            BOOL APIENTRY DllMain( HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)

            {
            switch (ul_reason_for_call)
            {
            case: DLL_PROCESS_ATTACH:
            break;
            case: DLL_THREAD_ATTACH:
            break;
            case: DLL_THREAD_DETACH:
            break;
            case: DLL_PROCESS_DETACH:
            break;
            return TRUE;
            }

             

            2.2 在dll中導出變量
            1、在dll中定義變量 extern int global;
            2、在.def中定義輸出 EXPORTS:
            global DATA
            3、 在應用程序中調用:#pragma comment(lib,"dllTest.lib")
            extern int global;
            注意在此引入的變量global,是一個地址,在使用時需要強制轉化為指針后再用,才能得到其值。
            (int *)global=10;
             

            在應用工程中引用DLL中全局變量的一個更好方法是:

            extern int _declspec(dllimport) global; //用_declspec(dllimport)導入
            通過_declspec(dllimport)方式導入的就是DLL中全局變量本身而不再是其地址了,建議在一切可能的情況下使用這種方式。

             

            2.3 dll導出類
            在定義的時候用

            class _declspec(dllexport) classname{

            }

             

            在類中引用的時候用加入類定義頭文件:

             

            #include “classname.h”

            Class _declspec(dllimport) classname 來導入類

             

            3、 MFC規則Dll
            MFC規則DLL的概念體現在兩方面:
               (1) 它是MFC的
               “是MFC的” 意味著可以在這種DLL的內部使用MFC;
            (2) 它是規則的
            “是規則的”意味著它不同于MFC擴展DLL,在MFC規則DLL的內部雖然可以使用MFC,但是其與應用程序的接口不能是MFC。而MFC擴展DLL與應用程序的接口 可以是MFC,可以從MFC擴展DLL中導出一個MFC類的派生類。
               Regular DLL能夠被所有支持DLL技術的語言所編寫的應用程序調用,當然也包括使用MFC的應用程序。在這種動態連接庫中,包含一個從CWinApp繼承下 來的類,DllMain函數則由MFC自動提供。

             

            (1)靜態鏈接到MFC 的規則DLL
               靜態鏈接到MFC的規則DLL與MFC庫(包括MFC擴展 DLL)靜態鏈接,將MFC庫的代碼直接生成在.dll文件中。在調用這種DLL的接口時,MFC使用DLL的資 源。因此,在靜態鏈接到MFC 的規則DLL中不需要進行模塊狀態的切換。使用這種方法生成的規則DLL其程序較大,也可能包含重復的代碼。
            (2)動態鏈接到MFC 的規則DLL
               動態鏈接到MFC 的規則DLL 可以和使用它的可執行文件同時動態鏈接到 MFC DLL 和任何MFC擴展 DLL。在使用了MFC共享庫的時候,默認情況下,MFC 使用主應用程序的資源句柄來加載資源模板。這樣,當DLL和應用程序中存在相同ID的資源時(即所謂的資源重復問題),系統可能不能獲得正確的資源。因此,對于共 享MFC DLL的規則DLL,我們必須進行模塊切換以使得MFC能夠找到正確的資源模板。

            我們可以在Visual C++中設置MFC規則DLL是靜態鏈接到MFC DLL還是動態鏈接到MFC DLL。如圖8,依次選擇Visual C++的project -> Settings -> General菜單或選項,在Microsoft Foundation Classes中進行設置。

             

            3.1規則DLL的創建;

            與非MFCdll不同的是,在其定義里面可以引入MFC類,其他與非MFC一樣
             

            3.2規則DLL的調用

            (1)顯示方式LoadLibrary , GetProcAdress , FreeLibrary

            (2)我們照樣可以在EXE程序中隱式調用MFC規則DLL,只需要將DLL工程生成的.lib文件和.dll文件拷入當前工程所在的目錄,并在cpp文件頂部添加:
            #pragma comment(lib,"RegularDll.lib")
             

            3.3共享MFC DLL的規則DLL的模塊切換

            應用程序進程本身及其調用的每個DLL模塊都具有一個全局唯一的HINSTANCE句柄,它們代表了DLL或EXE模塊在進程虛擬空間中的起始地址進程本身的模塊句柄一般為0x400000,而DLL模塊的缺省句柄為0x10000000。如果程序同時加載了多個DLL,則每個DLL模塊都會有不同的 HINSTANCE應用程序在加載DLL時對其進行了重定位。
              共享MFC DLL(或MFC擴展DLL)的規則DLL涉及到HINSTANCE句柄問題,HINSTANCE句柄對于加載資源特別重要。EXE和DLL都有其自己的資源,而且這些資源的ID可能重復,應用程序需要通過資源模塊的切換來找到正確的資源。如果應用程序需要來自于DLL的資源,就應將資源模塊句柄指定為 DLL的模塊句柄;如果需要EXE文件中包含的資源,就應將資源模塊句柄指定為EXE的模塊句柄。

             

            模塊的切換有三種方式:
            (1)在DLL函數中調用:AFX_MANAGE_STATE(AfxGetStaticModuleState());(推薦使用,最簡單)
            void ShowDlg(void)
            {
            //方法1:在函數開始處變更,在函數結束時恢復
            //將AFX_MANAGE_STATE(AfxGetStaticModuleState());作為接口函數的第一//條語句進行模塊狀態切換
            AFX_MANAGE_STATE(AfxGetStaticModuleState());
            CDialog dlg(IDD_DLL_DIALOG);//打開ID為2000的對話框
            dlg.DoModal();
            }
            (2)在DLL函數中調用AfxGetResourceHandle();
            AfxSetResourceHandle(HINSTANCE xxx);
            (3)由應用程序自身切換(不推薦,最麻煩)
             

            4、擴展MFCDLL

            MFC擴展DLL的內涵為MFC的擴展,用戶使用MFC擴展DLL就像使用MFC本身的DLL一樣。除了可以在MFC擴展DLL的內部使用MFC以外, MFC擴展DLL與應用程序的接口部分也可以是MFC。我們一般使用MFC擴展DLL來包含一些MFC的增強功能,譬如擴展MFC的CStatic、 CButton等類使之具備更強大的能力。
            導出一個類,直接在類聲明頭文件中使用AFX_EXT_CLASS即可,最后別忘了在調用dll的程序中加入class的頭文件

             

            5、總結:
            綜上所述:以上幾種dll主要由以下幾種區別:
             

            1、動態鏈接庫是將exe程序在程序執行的時候動態加載的,而靜態鏈接庫是在編譯的時 將其編譯在代碼之中的

            2、動態鏈接庫可以輸出變量、函數和類。其中每種輸出的方式與調用方式不盡相同:

             

            (1)變量:在dll中定義 extern int global;
            在.def文件中輸出 EXPORTS
            global DATA
            或extern _declspec(dllexport)int global(不用輸出文件了)
            在程序中調用:
            靜態調用:

            #pragma comment(lib,"dllTest.lib")

            extern int _declspec(dllimport) global;
            動態調用:
             

            (2)函數:在dll中定義

            extern “C” __declspec(dllexport) int add(int a, int b);
            也可以在.def文件中輸出該函數
            EXPORTS
            add @ 1
            在程序中調用:
            靜態調用:
            #pragma comment(lib,"dllTest.lib")
            extern "C" __declspec(dllimport) add(int x,int y);
            動態調用:
            typedef int(*lpAddFun)(int, int); //宏定義函數指針類型
            lpAddFun add;//函數指針
            HINSTANCE hDll=LoadLibrary(“path”);
            add=(lpAddFun)GetProcAddress(hDll, "add");//獲得dll中的add函數指針
            FreeLibrary(hDll);
            在從dll調用中返回的函數、指針或者類都是以指針的方式會的,即返回的是函數、變量或類的地址。這里一定要注意,不能簡單的用函數名來賦值。
            (3)類:在dll中定義:
            在定義的時候用 class _declspec(dllexport) classname{
            }
            在類中引用的時候用
            加入類定義頭文件:#include “classname.h”
            Class _declspec(dllimport) classname 來導入類
            3、除了擴展MFC的dll外,其他的dll均可被其他語言編寫的應用程序來調用。

             

            posted on 2012-05-09 14:34 Tim 閱讀(2781) 評論(0)  編輯 收藏 引用 所屬分類: 資源共享 、C/C++語言

            <2012年5月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            導航

            統計

            公告

            本博客原創文章,歡迎轉載和交流。不過請注明以下信息:
            作者:TimWu
            郵箱:timfly@yeah.net
            來源:www.shnenglu.com/Tim
            感謝您對我的支持!

            留言簿(9)

            隨筆分類(173)

            IT

            Life

            搜索

            積分與排名

            最新隨筆

            最新評論

            閱讀排行榜

            色综合久久夜色精品国产| 精品999久久久久久中文字幕| 四虎国产精品免费久久5151| 国产精品美女久久久久网| 色综合久久中文综合网| 久久国产乱子伦精品免费午夜| 久久本道久久综合伊人| 一级做a爰片久久毛片免费陪| 伊人久久大香线蕉av不变影院| 精品久久久噜噜噜久久久| 久久国产成人亚洲精品影院| 久久婷婷五月综合97色直播| 精品久久人妻av中文字幕| 思思久久99热免费精品6| 久久精品国产久精国产思思| 久久精品国产亚洲av瑜伽| 久久精品国产亚洲AV影院| 久久久久中文字幕| 婷婷久久香蕉五月综合加勒比| 国产成人AV综合久久| 久久久久久亚洲Av无码精品专口| 久久精品综合一区二区三区| 久久久久99精品成人片试看| 激情久久久久久久久久| 久久婷婷国产综合精品| 久久精品国产亚洲7777| 久久青青草原国产精品免费| 亚洲AV无码成人网站久久精品大| 久久久亚洲精品蜜桃臀| 91久久香蕉国产熟女线看| 少妇久久久久久久久久| 伊人久久大香线蕉成人| 久久久WWW成人免费毛片| 国产91久久精品一区二区| 午夜精品久久久久久久| 狠狠色狠狠色综合久久| 国产欧美久久久精品影院| 久久综合给合综合久久| 久久精品国产99久久久香蕉| 91精品观看91久久久久久 | 99久久无码一区人妻a黑|