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

            国产ww久久久久久久久久| 久久久久亚洲国产| 久久Av无码精品人妻系列| 久久只有这精品99| 亚洲欧洲中文日韩久久AV乱码| 久久有码中文字幕| 狠狠色婷婷久久综合频道日韩 | 亚洲精品无码久久千人斩| 久久91精品国产91| 欧美丰满熟妇BBB久久久| 亚洲熟妇无码另类久久久| 久久久亚洲裙底偷窥综合| 久久久亚洲欧洲日产国码aⅴ| 国产一区二区三区久久精品| 久久精品国产精品亚洲艾草网美妙| 久久e热在这里只有国产中文精品99| 久久精品亚洲欧美日韩久久| 亚洲欧美成人久久综合中文网| 久久婷婷五月综合97色直播| 久久综合国产乱子伦精品免费| 国产成人精品久久一区二区三区| 热久久这里只有精品| 香蕉久久永久视频| 国产成人精品免费久久久久| 精品国产青草久久久久福利| 狠狠色丁香久久婷婷综合图片| 东京热TOKYO综合久久精品| 久久婷婷五月综合97色直播| 国产偷久久久精品专区 | 久久久艹| 久久青青草原精品国产| 久久久久这里只有精品| 久久久久久午夜成人影院| 久久久久国产| 国内精品久久久久影院日本| 日批日出水久久亚洲精品tv| 久久久久久久97| 久久笫一福利免费导航| 午夜不卡888久久| 色诱久久久久综合网ywww| 亚洲精品国精品久久99热|