• <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>

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運轉,開心的工作
            簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            [DLL - Beginers]使用DEF文件修復函數名——對《使用LoadLibrary調用從Dll中輸出的class》的一點補充 收藏
            使用DEF文件修復函數名
            ——對《使用LoadLibrary調用從Dll中輸出的class》的一點補充

            作者 李成竹

            在《使用……》一文中,作者在“代碼”的第三點提到了“使用一個DEF文件修復了函數名”,但是并沒有講解什么是DEF文件,也沒有說明應該如何修復,可能會使某些初學者(包括我自己)感到疑惑。我也上網搜索了一下,講解DEF文件作用以及詳細使用方法的文章不多且比較零散,本文在此用一個簡單例子簡單闡述一下DEF文件一般的使用方法,以方便需要者查閱。

            ?

            DEF文件的全稱是Module-Definition File,即模塊定義文件,是用來定義EXE和DLL文件的一種文件格式,以文本形式保存(可用記事本創建/編輯)。由于鏈接器為大多數模塊定義聲明提供了對應的命令行選項,所以一般的Win32程序并不需要.DEF文件。但是在編寫DLL時,尤其是在編寫C++的DLL時,(由于名稱修飾)DEF文件還是有它的用武之地的。

            ※注:關于“名稱修飾”在很多地方都有介紹,文中不作講解。

            ?

            DEF文件的主要內容是由一系列的聲明(statement)組成,包括NAME、LIBRARY、DISCRIPTION、STACKSIZE、SECTIONS、EXPORTS、VERSION。

            ?

            ¨???????? NAME:指定輸出文件的文件名,設置image基址

            ¨???????? LIBRARY(DLL):指定DLL的內部名稱和加載時的基址

            ¨???????? DISCRIPTION:文件描述

            ¨???????? STACKSIZE:設置棧的大小

            ¨???????? SECTIONS:設置image文件的一個或多個段屬性

            ¨???????? EXPORTS:定義輸出列表

            ¨???????? VERSION:指定文件版本

            ?

            其中最常用的是LIBRARY、EXPORTS和DISCRIPTION。

            ?

            示例

            1.????? 現在有一個已經編寫好的類要用DLL輸出,并通過函數名對DLL進行動態調用。其頭文件和源文件如下:

            Header File:

            #ifdef LIBDLL_EXPORTS
            #define LIBDLL_API __declspec(dllexport)
            #else
            #define LIBDLL_API __declspec(dllimport)
            #endif

            #include <iostream.h>

            // This class is exported from the LibDll.dll
            class LIBDLL_API CTest
            {
            ? int data;

            public:
            ? CTest();
            ? void print();
            };

            Source File:

            #include "stdafx.h"
            #include "LibDll.h"

            BOOL APIENTRY DllMain( HANDLE hModule,
            ?????????????????????? DWORD? ul_reason_for_call,
            ?????????????????????? LPVOID lpReserved
            ???????????????????????????????? )
            {
            ??? switch (ul_reason_for_call)
            ? {
            ???????? case DLL_PROCESS_ATTACH:
            ???????? case DLL_THREAD_ATTACH:
            ???????? case DLL_THREAD_DETACH:
            ???????? case DLL_PROCESS_DETACH:
            ???????????????? break;
            ??? }
            ??? return TRUE;
            }

            CTest::CTest()
            {
            ? this->data = 0;
            }

            void CTest::print()
            {
            ? cout<<"The member function print() is from a DLL.\n";
            }


            從代碼中可以看到,DLL中定義了一個CTest類,它有一個構造函數和一個成員函數print()。

            2.????? 然后新建一個Win32 Console Application來調用DLL。

            Header File:

            #define LIBDLL_API __declspec(dllimport)

            #include <iostream.h>

            // This class is exported from the LibDll.dll
            class LIBDLL_API CTest
            {
            ? int data;

            public:
            ? CTest();
            ? void print();
            };

            Source File:

            #include "stdafx.h"
            #include <iostream.h>
            #include <malloc.h>
            #include <windows.h>
            #include "LibDll.h"

            typedef void (WINAPI *PCTOR)();
            typedef void (*PPRINT)();

            inline void CTest_print(HMODULE, CTest*);
            inline void CTest_CTest(HMODULE, CTest*);

            int main(int argc, char* argv[])
            {
            ? //加載DLL
            ? HMODULE hmod = LoadLibrary("LibDll.dll");
            ? if(hmod == NULL)
            ? {
            ???????? cout<<"Failed loading DLL.\n";

            ???????? return 1;
            ? }

            ? //創建類對象
            ? CTest* pCTest = (CTest*)malloc(sizeof(CTest));

            ? //初始化CTest對象
            ? CTest_CTest(hmod, pCTest);
            ????????
            ?//調用成員函數
            ? CTest_print(hmod, pCTest);

            ? FreeLibrary(hmod);
            ? free(pCTest);

            ? cout<<"Press [Enter] to exit.";
            ? cin.peek();

            ? return 0;
            }//end main


            void CTest_print(HMODULE hMod, CTest* pObj)
            {
            ? PPRINT pprint = (PPRINT)GetProcAddress(hMod, "print");
            ? if(pprint == NULL)
            ? {
            ???????? cout<<"Function print() not found.\n";
            ? }
            ? else
            ? {
            ???????? __asm{ MOV ECX, pObj}

            ???????? pprint();
            ? }
            }

            void CTest_CTest(HMODULE hMod, CTest* pObj)
            {
            ? PCTOR pCtor = (PCTOR)GetProcAddress(hMod, "CTest");
            ? if(pCtor == NULL)
            ? {
            ???????? cout<<"Function CTest() not found.\n";
            ? }
            ? else
            ? {
            ???????? __asm{ MOV ECX, pObj}

            ???????? pCtor();
            ? }
            }

            ?

            本來到這里就應該可以正常運行了,但是你會發現在執行CTest_CTest函數時會提示"Function CTest() not found."。為什么會找不到函數呢?那是因為C++編譯器在生成DLL時對輸出函數的名稱進行來“修飾”,所以DLL中的函數名稱已經不再是我們在代碼中所寫的函數名,這個時候就需要用DEF文件來進行“函數名稱修復”。

            3.????? 用Dumpbin的/EXPORTS參數打開LibDll.Dll,截圖如下:

            ?

            其中1和3就是CTest類的構造函數和print()函數的實際名稱(嚇人吧……)。然后我們在DLL的工程目錄中新建一個“LibDll.def”文件,并在“工程->設置->Link”中添加參數(/def:".\libdll.def"),并編

            輯DEF文件內容如下:


            LIBRARY LibDll
            EXPORTS
            ?CTest?? = ??0CTest@@QAE@XZ
            ?print?? = ?print@CTest@@QAEXXZ


            相信你已經看出來了,這實際上是一個函數名映射。

            再用Dumpbin打重新編譯得到的DLL文件,截圖如下:


            圖中的4和5就是修復后的函數名。

            現在再運行第二步中的程序就可以成功地調用DLL里的函數了!

            尾注


            文中只使用了DEF文件的一小部分功能,詳細資料請參見MSDN。

            按照MSDN上的說法,有三種方法可以用來輸出函數,按推薦順序如下:

            在源代碼中使用__declspec(dllexport)關鍵字(調用工程需包含*.lib)
            使用DEF文件中的EXPORTS聲明(不需要*.lib,可實現動態調用)
            在LINK命令中使用/EXPORT參數(效果和DEF文件相同)
            具體應該使用哪種方法,還應該視用途由使用者自己決定。

            ?

            作者水平有限,文中難免不當之處,歡迎大家指出和批評。


            本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/jdcb2001/archive/2006/11/21/1401569.aspx

            99久久精品影院老鸭窝| 亚洲国产成人久久一区久久| 婷婷久久综合九色综合绿巨人| 精品久久久噜噜噜久久久| 四虎影视久久久免费| 国产成人久久777777| 久久不射电影网| 成人久久综合网| 粉嫩小泬无遮挡久久久久久| 亚洲AV无一区二区三区久久| 中文字幕无码免费久久| 久久人人爽人人爽人人片av麻烦 | 99麻豆久久久国产精品免费| 精品国产乱码久久久久软件 | 区久久AAA片69亚洲| 久久久久国产亚洲AV麻豆| 狠狠色综合网站久久久久久久| 久久97精品久久久久久久不卡| 国产精品无码久久综合| 久久99精品综合国产首页| 久久免费美女视频| 狠狠色丁香婷婷综合久久来来去 | 99久久精品国产一区二区| 欧美亚洲国产精品久久| 99久久国产宗和精品1上映 | 久久精品蜜芽亚洲国产AV| 1000部精品久久久久久久久| 久久无码av三级| 久久无码一区二区三区少妇 | 久久国产热精品波多野结衣AV| 99久久精品午夜一区二区| 香蕉久久夜色精品国产小说| 久久久久久A亚洲欧洲AV冫| 久久精品国产亚洲αv忘忧草| 久久天天躁狠狠躁夜夜avapp| 国产精品天天影视久久综合网| 狠狠人妻久久久久久综合蜜桃| 亚洲欧洲精品成人久久奇米网| 久久久久久久97| 久久久精品日本一区二区三区| 亚洲国产一成人久久精品|