• <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.¢%

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

            [DLL - Beginers]使用DEF文件修復(fù)函數(shù)名——對《使用LoadLibrary調(diào)用從Dll中輸出的class》的一點(diǎn)補(bǔ)充 收藏
            使用DEF文件修復(fù)函數(shù)名
            ——對《使用LoadLibrary調(diào)用從Dll中輸出的class》的一點(diǎn)補(bǔ)充

            作者 李成竹

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

            ?

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

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

            ?

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

            ?

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

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

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

            ¨???????? STACKSIZE:設(shè)置棧的大小

            ¨???????? SECTIONS:設(shè)置image文件的一個(gè)或多個(gè)段屬性

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

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

            ?

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

            ?

            示例

            1.????? 現(xiàn)在有一個(gè)已經(jīng)編寫好的類要用DLL輸出,并通過函數(shù)名對DLL進(jìn)行動態(tài)調(diào)用。其頭文件和源文件如下:

            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中定義了一個(gè)CTest類,它有一個(gè)構(gòu)造函數(shù)和一個(gè)成員函數(shù)print()。

            2.????? 然后新建一個(gè)Win32 Console Application來調(diào)用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;
            ? }

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

            ? //初始化CTest對象
            ? CTest_CTest(hmod, pCTest);
            ????????
            ?//調(diào)用成員函數(shù)
            ? 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();
            ? }
            }

            ?

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

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

            ?

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

            輯DEF文件內(nèi)容如下:


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


            相信你已經(jīng)看出來了,這實(shí)際上是一個(gè)函數(shù)名映射。

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


            圖中的4和5就是修復(fù)后的函數(shù)名。

            現(xiàn)在再運(yùn)行第二步中的程序就可以成功地調(diào)用DLL里的函數(shù)了!

            尾注


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

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

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

            ?

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


            本文來自CSDN博客,轉(zhuǎn)載請標(biāo)明出處:http://blog.csdn.net/jdcb2001/archive/2006/11/21/1401569.aspx

            欧美黑人激情性久久| 久久91这里精品国产2020| 女人高潮久久久叫人喷水| 色欲综合久久躁天天躁| 色综合久久无码五十路人妻| 国产成人久久精品二区三区| 久久精品免费一区二区| 久久青草国产手机看片福利盒子| 色诱久久av| 国产精品99久久久久久董美香| 久久热这里只有精品在线观看| 久久成人国产精品二三区| 亚洲精品无码久久久久久| 久久国产香蕉一区精品| 久久大香香蕉国产| 久久精品亚洲乱码伦伦中文| 99re久久精品国产首页2020| 四虎影视久久久免费观看| 久久99精品国产| 久久久久久久久无码精品亚洲日韩 | 久久嫩草影院免费看夜色| 亚洲国产美女精品久久久久∴| 久久婷婷色综合一区二区| 久久精品国产一区二区三区日韩| 99久久国产宗和精品1上映| 开心久久婷婷综合中文字幕| 久久国产成人精品麻豆| 国产精品免费福利久久| 久久久av波多野一区二区| 18岁日韩内射颜射午夜久久成人| 久久久久久毛片免费看| 国产成人久久精品二区三区| 久久综合九色综合久99| 夜夜亚洲天天久久| 99久久亚洲综合精品成人| 久久免费精品视频| 国内精品伊人久久久久影院对白 | 97久久精品国产精品青草| 国产产无码乱码精品久久鸭| 99久久免费国产精精品| 91精品婷婷国产综合久久|