C++類(lèi)包含函數(shù)說(shuō)明
C++類(lèi)包含三個(gè)我們關(guān)心的函數(shù):構(gòu)造函數(shù),析構(gòu)函數(shù),和所有重要的 DoSomething 函數(shù),我們需要把每一個(gè)函數(shù)包裹成與其等價(jià)的C++函數(shù),在這里拿出來(lái)和大家分享一下。
- // original class
- class CFoo
- {
- public:
- CFoo(int x);
- ~CFoo();
- int DoSomething(int y);
- };
- // flattened C code
- void* __stdcall new_CFoo(int x)
- {
- return new CFoo(x);
- }
- int __stdcall CFoo_DoSomething(void* handle, int y)
- {
- CFoo *foo = reinterpret_cast<CFoo *>(handle);
- return foo->DoSomething(y);
- }
- void __stdcall delete_CFoo(void *handle)
- {
- CFoo *foo = reinterpret_cast<CFoo *>(handle);
- delete foo;
- }
這里有幾個(gè)比較重要的地方要注意。首先,注意每一個(gè)C++類(lèi)被映射為一個(gè)簡(jiǎn)單的 C 函數(shù)。其次,觀察到我們?yōu)?C 函數(shù)明確地使用 __stdcall 調(diào)用習(xí)慣。在前一篇 DLL 文章里,我們知道簡(jiǎn)單的調(diào)用在 MSVC DLL 里的無(wú)格式 C 函數(shù),真是很麻煩。
如果我們放棄越過(guò)種種艱難困苦去用它,我們可以使這個(gè)努力稍微容易一點(diǎn)。讓 Borland 調(diào)用 Microsoft DLL 最簡(jiǎn)單的辦法是 DLL 導(dǎo)出無(wú)格式,無(wú)修飾,__stdcall 調(diào)用習(xí)慣的 C++函數(shù)。Borland 和 Microsoft 對(duì) __cdecl 函數(shù)的處理上是不同的。
通常,他們對(duì) __stdcall 函數(shù)也不同,因?yàn)?MSVC 修飾 __stdcall 函數(shù),但我們可以通過(guò)添加一個(gè) DEF 文件到 MSVC 工程里來(lái)阻止這種行為。參見(jiàn)下載部分的例子有 DEF 文件的例子。其它關(guān)于代碼要注意的事情是 new_CFoo 函數(shù)返回一個(gè)指向 CFoo 對(duì)象的指針。BCB 調(diào)用者必須在本地保存這個(gè)指針。這可能看起來(lái)和這篇文章的主題有點(diǎn)矛盾。
畢竟,我想 BCB 不能使用來(lái)自 MSVC DLL 的 C++類(lèi)?如果那是正確的,那么為什么我們還要返回一個(gè) CFoo 對(duì)象指針呢?答案是 BCB 不能調(diào)用 MSVC DLL 導(dǎo)出類(lèi)的成員函數(shù)。但是,這并不意味著它不能存儲(chǔ)這樣對(duì)象的地址。new_CFoo 返回的是一個(gè) CFoo 對(duì)象的指針。
BCB 客戶端可以存儲(chǔ)這個(gè)指針,但不能用。BCB 不能廢棄它(不應(yīng)當(dāng)嘗試這么做)。讓這個(gè)觀點(diǎn)更容易理解一點(diǎn),new_CFoo 返回一個(gè)空指針(總之它不能返回別的什么東西)。在 BCB 這邊,除了存儲(chǔ)它,然后把它傳回給 DLL,沒(méi)有什么可以安全地處理這個(gè)空指針的方法。
Ok,在我們繼續(xù)前進(jìn)之前,還有另外兩個(gè)要十分注意的地方。首先,注意 CFoo_DoSomething 把空指針作為它的***個(gè)參數(shù)這個(gè)空指針與 new_CFoo 返回的是同一個(gè)空指針??罩羔樣?reinterpret_cast 被追溯到 CFoo 對(duì)象(你知道,當(dāng)你看到一個(gè) reinterpret_cast 的時(shí)候。
你正在處理是難看的代碼)。DoSomething 成員函數(shù)在轉(zhuǎn)換之后被調(diào)用。***注意空指針也是C++類(lèi)的參數(shù)。包裝 DLL 刪除對(duì)象是至關(guān)緊要的。你不應(yīng)當(dāng)在 BCB 里對(duì)空指針調(diào)用 delete。顯然它不會(huì)按你想的去做。
【編輯推薦】