DLL入門(mén)淺析(4)——從DLL中導(dǎo)出類(lèi)
DLL頭文件:























DLL實(shí)現(xiàn)文件:

































應(yīng)用程序調(diào)用DLL














大家可能發(fā)現(xiàn)了,上面我沒(méi)有使用模塊定義文件(.def)聲明導(dǎo)出類(lèi)也沒(méi)有用顯式鏈接導(dǎo)入DLL。
用Depends查看前面編譯出來(lái)的DLL文件,會(huì)發(fā)現(xiàn)里面導(dǎo)出了很奇怪的symbol,這是因?yàn)镃++編譯器在編譯時(shí)會(huì)對(duì)symbol進(jìn)行修飾。
這是我從別人那兒轉(zhuǎn)來(lái)的截圖。
網(wǎng)上找了下,發(fā)現(xiàn)了C++編譯時(shí)函數(shù)名的修飾約定規(guī)則
__stdcall調(diào)用約定:
1、以"?"標(biāo)識(shí)函數(shù)名的開(kāi)始,后跟函數(shù)名;
2、函數(shù)名后面以"@@YG"標(biāo)識(shí)參數(shù)表的開(kāi)始,后跟參數(shù)表;
3、參數(shù)表以代號(hào)表示:
X——void,
D——char,
E——unsigned char,
F——short,
H——int,
I——unsigned int,
J——long,
K——unsigned long,
M——float,
N——double,
_N——bool,
....
PA——表示指針,后面的代號(hào)表明指針類(lèi)型,如果相同類(lèi)型的指針連續(xù)出現(xiàn),以"0"代替,一個(gè)"0"代表一次重復(fù);
4、參數(shù)表的第一項(xiàng)為該函數(shù)的返回值類(lèi)型,其后依次為參數(shù)的數(shù)據(jù)類(lèi)型,指針標(biāo)識(shí)在其所指數(shù)據(jù)類(lèi)型前;
5、參數(shù)表后以"@Z"標(biāo)識(shí)整個(gè)名字的結(jié)束,如果該函數(shù)無(wú)參數(shù),則以"Z"標(biāo)識(shí)結(jié)束。
其格式為"?functionname@@YG*****@Z"或?functionname@@YG*XZ,
int Test1(char *var1,unsigned long)-----“?Test1@@YGHPADK@Z” void Test2() -----“?Test2@@YGXXZ”
__cdecl調(diào)用約定:
規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開(kāi)始標(biāo)識(shí)由上面的"@@YG"變?yōu)?@@YA"。
__fastcall調(diào)用約定:
規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開(kāi)始標(biāo)識(shí)由上面的"@@YG"變?yōu)?@@YI"。
VC++對(duì)函數(shù)的省缺聲明是"__cedcl",將只能被C/C++調(diào)用。
雖然因?yàn)镃++編譯器對(duì)symbol進(jìn)行修飾的原因不能直接用def文件聲明導(dǎo)出類(lèi)和顯式鏈接,但是可以用另外一種取巧的方式。
friend DLLClass* CreatDLLClass();
然后聲明CreatDLLClass()為導(dǎo)出函數(shù),通過(guò)調(diào)用該函數(shù)返回一個(gè)DLLClass類(lèi)的對(duì)象,同樣達(dá)到了導(dǎo)出類(lèi)的目的。
這樣,就可以用顯式鏈接來(lái)調(diào)用CreatDLLClass(),從而得到類(lèi)對(duì)象了。
posted on 2009-07-20 20:50 Saga 閱讀(50569) 評(píng)論(8) 編輯 收藏 引用 所屬分類(lèi): Windows