• <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>
            白駒過(guò)隙
            學(xué)而不思則罔,思而不學(xué)則怠
            posts - 4,  comments - 3,  trackbacks - 0

            前一篇文章中是用隱式方法調(diào)用 DLL 的。下面介紹顯式調(diào)用。

            顯式的調(diào)用就是指在應(yīng)用程序中用 LoadLibrary MFC 提供的 AfxLoadLibrary 顯式的將自己所做的動(dòng)態(tài)連接庫(kù)調(diào)進(jìn)來(lái),動(dòng)態(tài)連接庫(kù)的文件名即是上面兩個(gè)函數(shù)的參數(shù),再用 GetProcAddress() 獲取想要引入的函數(shù)。

            ?

            在前面的基礎(chǔ)上再建一個(gè)測(cè)試工程,代碼如下:

            #include "stdafx.h"

            #include <iostream>

            //#include "Try.h"

            using namespace std;

            ?

            typedef int (*Fn_FunType)() ;

            ?

            int main()

            {

            ?

            ??? HINSTANCE hInstance = 0;

            ??? hInstance = LoadLibrary("Try.dll") ;

            ??? if (hInstance)

            ??? {

            ?????? Fn_FunType pFun = (Fn_FunType)::GetProcAddress(hInstance ,"fnTry") ;

            ?????? if (pFun)

            ?????? {

            ?????????? cout<<pFun()<<endl;

            ?????? }

            ??? }

            ??? int n = 10 ;

            n = *(int *)::GetProcAddress(hInstance ,"nTry") ;

            ??? cout<<n<<endl;

            ??? system("pause") ;

            ??? return 0 ;

            }

            ?

            ? 上面的代碼貌似沒(méi)上面問(wèn)題,但是實(shí)際上第一個(gè) cout 永遠(yuǎn)也不會(huì)執(zhí)行,第二個(gè) cout 輸出的是 10 。為什么會(huì)這樣?

            先來(lái)看看,上面的代碼在 DLL 中是個(gè)什么模樣,用 VS 自帶的 dependency walker 可以查看 DLL 文件,該工具在 ? Microsoft Visual Studio 8/ common7/tools/bin 下。如下圖所示:

            ?



            ?? 原來(lái),
            "C" 或者 "C++" 函數(shù)在編譯器內(nèi)部(編譯和鏈接)通過(guò)修飾名識(shí)別。修飾名是編譯器在編譯函數(shù)定義或者原型時(shí)生成的字符串。有些情況下使用函數(shù)的修飾名是必要的,如在模塊定義文件里頭指定輸出 "C++" 重載函數(shù)、構(gòu)造函數(shù)、析構(gòu)函數(shù),又如在匯編代碼里調(diào)

            "C"" "C++" 函數(shù)等。關(guān)于修飾名后面會(huì)講到。

            ?

            將上面的代碼稍微作一下修改:

            ?????? Fn_FunType pFun = (Fn_FunType)::GetProcAddress(hInstance ," ?fnTry@@YAHXZ") ;

            int n = *(int *)::GetProcAddress(hInstance ,"?nTry@@3HA") ;

            Ok, 現(xiàn)在可以正常輸出了。

            ?

            ?

            上面的方法可以導(dǎo)出全局函數(shù)和變量,但是卻無(wú)法導(dǎo)出成員函數(shù)和成員變量。如下:

            typedef void (CTry::*Fn_FooType)() ;

            int main()

            {

            ?

            ??? HINSTANCE hInstance = 0;

            ??? hInstance = LoadLibrary("Try.dll") ;

            ??? CTry *pTry = (CTry *)_alloca(sizeof(CTry));

            ??? Fn_FooType pFun = (Fn_FooType)::GetProcAddress(hInstance, "?print@CTry@@QAEXXZ") ;
            (pTry->*pFun)() ;

            Return 0 ;
            }

            ?

            編譯器會(huì)提示, FARPROC 類型不能轉(zhuǎn)換成 Fn_FooType 類型。因?yàn)?/span> C++ 中,成員函數(shù)的指針是個(gè)比較特殊的東西。對(duì)普通的函數(shù)指針來(lái)說(shuō),可以視為一個(gè)地址 , 在需要的時(shí)候可以任意轉(zhuǎn)換并直接調(diào)用。但對(duì)成員函數(shù)來(lái)說(shuō),調(diào)用的時(shí)候也必須采用特殊的語(yǔ)法。 C++ 專門為成員指針準(zhǔn)備了三個(gè)運(yùn)算符 : "::*" 用于指針的聲明,而 "->*" ".*" 用來(lái)調(diào)用指針指向的函數(shù)。而且對(duì)成員函數(shù)指針不能進(jìn)行類型轉(zhuǎn)換。簡(jiǎn)單的說(shuō),每個(gè)成員函數(shù)指針都是一個(gè)獨(dú)有的類型,無(wú)法轉(zhuǎn)換到任何其它類型。

            ?

            但是在 C++ 中,總是有辦法的。 使用 union 類型可以逃避 C++ 的類型轉(zhuǎn)換檢測(cè)。為了通用,使用模板傳入數(shù)據(jù)類型。

            template <classDest, classSrc>

            Dest force_cast(Srcsrc)

            {

            ??? union

            ??? {

            ?????? Dest d;

            ?????? Src s;

            ??? } convertor;

            ?

            ??? convertor.s = src;

            ??? return convertor.d;

            }

            ?

            上面的代碼變?yōu)椋?/span>

            typedef void (CTry::*Fn_FooType)() ;

            ?

            int main()

            {

            ?

            ??? HINSTANCE hInstance = 0;

            ??? hInstance = LoadLibrary("Try.dll") ;

            ??? if (hInstance)

            ??? {

            ??????

            ?????? CTry *pTry = (CTry *)_alloca(sizeof(CTry));

            ?

            ?????? FARPROC fp ? = ::GetProcAddress(hInstance ,"?print@CTry@@QAEXXZ") ;

            ?????? Fn_FooType pFun = force_cast<Fn_FooType>(fp) ;

            ?????? if (pFun)

            ?????? {

            ?????????? (pTry->*pFun)() ;

            ?????? }

            ??? }

            ??? return 0 ;

            }

            ?

            posted on 2009-03-06 00:54 隙中駒 閱讀(352) 評(píng)論(1)  編輯 收藏 引用

            FeedBack:
            # re: DLL學(xué)習(xí)筆記2
            2009-03-12 09:30 | guest
            圖片的路徑是“e:/1.bmp”,你本地應(yīng)該可以顯示,但是其他地方都看不到圖了。
            請(qǐng)修改下圖片鏈接。  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理



            <2025年7月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿(2)

            隨筆檔案(4)

            文章分類(1)

            文章檔案(1)

            友情鏈接

            最新隨筆

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久狠狠色狠狠色综合| 久久亚洲精品国产精品婷婷| 欧美噜噜久久久XXX| …久久精品99久久香蕉国产| 情人伊人久久综合亚洲| 亚洲欧美日韩精品久久亚洲区| 亚洲国产天堂久久综合| 97久久超碰国产精品旧版| 久久精品视频91| 久久国产精品一区二区| 久久成人小视频| 99久久国产综合精品五月天喷水| 久久高潮一级毛片免费| 久久精品国产亚洲AV香蕉| 久久久久婷婷| 99久久精品无码一区二区毛片 | 99久久久国产精品免费无卡顿| 久久精品国产亚洲沈樵| 伊人久久大香线蕉亚洲| 久久久久久一区国产精品| 久久国产精品国产自线拍免费 | 久久影视综合亚洲| 99久久精品免费观看国产| 精品久久久久久国产潘金莲| 无码八A片人妻少妇久久| 久久九九久精品国产| 成人精品一区二区久久久| 久久99精品久久久久久久不卡| 无码八A片人妻少妇久久| 久久精品国产第一区二区| 国产毛片久久久久久国产毛片| 久久本道伊人久久| 国产亚洲欧美成人久久片| 亚洲AV日韩精品久久久久久| 久久妇女高潮几次MBA| 2019久久久高清456| 久久只有这里有精品4| 国产成人精品综合久久久久| 亚洲国产精品久久久天堂| 亚洲欧洲日产国码无码久久99| 亚洲∧v久久久无码精品|