• <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),開(kāi)心的工作
            簡(jiǎn)單、開(kāi)放、平等的公司文化;尊重個(gè)性、自由與個(gè)人價(jià)值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            總結(jié)一下DLL聲明的一些問(wèn)題

            Posted on 2009-09-04 09:25 S.l.e!ep.¢% 閱讀(670) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): DLL
            有關(guān)DLL的問(wèn)題現(xiàn)在資料很多,但是很多人寫(xiě)DLL時(shí)經(jīng)常出現(xiàn)調(diào)用程序無(wú)法找到相關(guān)的導(dǎo)出函數(shù)的問(wèn)題,這里主要的原因是DLL在聲明時(shí)出的問(wèn)題。
            在這里主要有兩個(gè)問(wèn)題,一個(gè)是調(diào)用約定的問(wèn)題,一個(gè)是函數(shù)名修飾的問(wèn)題,而這兩個(gè)問(wèn)題又是相互影響的。
            一:聲明為:extern "C" int __declspec(dllexport)add(int x, int y);
            這種聲明是強(qiáng)制用C語(yǔ)言方式進(jìn)行修飾,且用C的默認(rèn)約定,即__cdecl方式。這種方式編譯產(chǎn)生的DLL中有一個(gè)導(dǎo)出函數(shù):add,不加任何修飾。
            二:聲明為:extern "C" int __declspec(dllexport) __stdcall add(int x, int y);
            這種聲明是強(qiáng)制用C語(yǔ)言方式進(jìn)行修飾,且用stdcall約定,這種方式編譯產(chǎn)生的DLL中有一個(gè)導(dǎo)出函數(shù):_add@8,即前面有“_”,后面加了參數(shù)長(zhǎng)。
            三:聲明為:int __declspec(dllexport) __stdcall add(int x, int y);
            這種聲明不強(qiáng)制用C語(yǔ)言方式進(jìn)行修飾,但是用stdcall約定,這種方式編譯產(chǎn)生的DLL中有一個(gè)導(dǎo)出函數(shù):?add@@YGHHH@Z。這個(gè)名字很怪,后面的不好理解。
            四:聲明為:int __declspec(dllexport) __cdecl add(int x, int y);
            這種聲明是不強(qiáng)制用C語(yǔ)言修飾,且用cdecl約定,這種方式編譯產(chǎn)生的DLL中有一個(gè)導(dǎo)出函數(shù):?add@@YAHHH@Z,注意看,和第三種方有一點(diǎn)不同。

            實(shí)驗(yàn)一:顯式調(diào)用方式調(diào)用DLL中的add函數(shù)。
            #include <stdio.h>
            #include <windows.h>
            typedef? int(_stdcall *lpAddFun)(int, int); //宏定義函數(shù)指針類(lèi)型
            int main(int argc, char *argv[])
            {
            HINSTANCE hDll; //DLL句柄
            lpAddFun addFun; //函數(shù)指針
            hDll = LoadLibrary("1.dll");
            if (hDll != NULL)
            {
            addFun = (lpAddFun)GetProcAddress(hDll, "add");
            if (addFun != NULL)
            {
            int result = addFun(2, 3);
            printf("%d", result);
            }
            else
            printf("No Function");
            }
            else
            printf("NO DLL");
            FreeLibrary(hDll);
            return 0;
            }
            方式一:調(diào)用成功。另外三種方式全部出錯(cuò)
            實(shí)驗(yàn)二:隱式調(diào)用DLL中的add函數(shù)
            #include <stdio.h>
            #include <windows.h>
            #pragma comment(lib,"1.lib")
            extern "C" int __declspec(dllimport) add(int x, int y);//聲明方式隨著DLL中的聲明方式改變
            int main(int argc, char *argv[])
            {
            int result = add(2, 3);
            printf("%d", result);
            return 0;
            }
            方式一:調(diào)用成功。另外發(fā)現(xiàn)一個(gè)奇怪現(xiàn)象:在調(diào)用程序中
            聲明函數(shù)時(shí)extern "C" int __declspec(dllimport) add(int x, int y);
            寫(xiě)作:extern "C" int __declspec(dllecprot) add(int x, int y);同樣成功,將__declspec(…)去掉也同樣成功。換句話說(shuō),在調(diào)用DLL的程序中,導(dǎo)入是沒(méi)有必要加的。
            方式二:調(diào)用成功。同樣出現(xiàn)上面導(dǎo)入標(biāo)識(shí)可以不加的現(xiàn)象。
            方式三:調(diào)用成功,同樣也出現(xiàn)上面導(dǎo)入標(biāo)識(shí)可以不加的現(xiàn)象。
            方式四:調(diào)用成功,同樣也出現(xiàn)上面導(dǎo)入標(biāo)識(shí)可以不加的現(xiàn)象。
            總結(jié):對(duì)于DLL導(dǎo)出函數(shù)聲明的四種寫(xiě)法,在動(dòng)態(tài)調(diào)用時(shí),
            聲明成這樣:extern "C" int __declspec(dllimport) add(int x, int y);是最好的,其它聲明方式調(diào)用都沒(méi)有成功。但是眾所周知,windows默認(rèn)的調(diào)用約定是stdcall方式,如果想別的語(yǔ)言能用DLL的話,最好是將調(diào)用約定寫(xiě)成stdcall方式,但是這種方式又不能動(dòng)態(tài)調(diào)用。
            在隱式調(diào)用時(shí),四種聲明方式都是可以的,只要調(diào)用者的聲明方式和DLL聲明時(shí)的方式一致即可。另外,在調(diào)用程序中對(duì)于導(dǎo)入的聲明是可以去掉的,大量書(shū)籍中關(guān)于導(dǎo)入、導(dǎo)出的問(wèn)題都是利用宏來(lái)處理的,如:在頭文件中寫(xiě)作:
            #ifdef DLL_FILE
            extern "C" int __declspec(dllexport) add(int x, int y);
            #else
            extern "C" int __declspec(dlleximport) add(int x, int y);
            這樣這個(gè)頭文件既可以用在DLL工程中,又可以用在調(diào)用程序中,但是經(jīng)過(guò)實(shí)驗(yàn)發(fā)現(xiàn),這個(gè)根本就沒(méi)有必要,在調(diào)用者程序中不管是寫(xiě)作__declspce(dllexport)還是寫(xiě)作__declspec(dllimport)或者不寫(xiě)都能成功調(diào)用。
            關(guān)于DEF文件
            在DLL工程中引用DEF文件,內(nèi)容如下:
            LIBRARY 1
            EXPORTS
            add @ 1
            通過(guò)depends查看導(dǎo)出函數(shù)全是add,但是隱式方式調(diào)用時(shí),還是要求調(diào)用者的聲明方式和DLL中聲明方式相同。
            對(duì)于動(dòng)態(tài)調(diào)用實(shí)驗(yàn)結(jié)果:
            方式一:成功。方式二:不成功,但是將函數(shù)指針改為typedef int(_stdcall *lpAddFun)(int, int);成功,即調(diào)用者要聲明約定方式與DLL中聲明的調(diào)用約定方式相同,否則報(bào)錯(cuò)。
            方式三:同方式二,同樣要將函數(shù)指針改為typedef int(_stdcall *lpAddFun)(int, int);才成功完成調(diào)用。
            方式四:成功。
            總結(jié):通過(guò)DEF文件來(lái)導(dǎo)出函數(shù),調(diào)用者同樣也要聲明相同的調(diào)用約定,即_stdcall或是_cdecl必須要相同,其中_cdecl是C語(yǔ)言默認(rèn)方式。
            久久精品国产精品青草app| 久久精品无码一区二区无码 | …久久精品99久久香蕉国产| 国产成人精品久久| 国产欧美一区二区久久| 国产—久久香蕉国产线看观看 | 久久人妻少妇嫩草AV无码蜜桃| 欧美亚洲日本久久精品| 久久精品久久久久观看99水蜜桃| 久久精品国产乱子伦| 久久综合中文字幕| 国产69精品久久久久观看软件| 久久婷婷五月综合97色| 91麻精品国产91久久久久| 久久国产欧美日韩精品免费| 精品免费久久久久久久| 久久国产免费| 国产精品久久精品| 热久久视久久精品18| 久久久国产精品亚洲一区| 精品久久久久久无码人妻热| 久久人人爽人人人人片av| 久久综合丁香激情久久| 国产A三级久久精品| segui久久国产精品| 久久人人妻人人爽人人爽| 国产精品VIDEOSSEX久久发布| 日本WV一本一道久久香蕉| 国内精品久久久久久久涩爱 | 欧美精品国产综合久久| 国产精品青草久久久久福利99| 久久精品国产亚洲av麻豆蜜芽| 久久成人精品| 久久亚洲国产精品一区二区| 久久成人国产精品免费软件| 国产一区二区精品久久凹凸| 国内精品久久久久久久97牛牛 | 99久久国产宗和精品1上映| 国产成人精品久久综合| 国产产无码乱码精品久久鸭 | 久久久久久久亚洲精品|