c++調(diào)用 c# COM:hr=0x80040154的常見問題
當(dāng)你使用tlb導(dǎo)入類型信息,并使用智能指針來生成對象的時(shí)候,經(jīng)常會碰見這么一個(gè)問題.0x80040154解釋是"沒有注冊類別",對組件經(jīng)過反復(fù)注冊,確信已經(jīng)成功注冊了,問題依舊.
原因分析:
#include "olectl.h"
[ uuid(7044762E-1832-4997-85CE-F8329F3F1A44), version(1.0) ]
library Fri
{
importlib("stdole32.tlb");
importlib("stdole2.tlb");
// One のプライマリ ディスパッチ インターフェイス
??
[ uuid(77A9625E-E129-4465-AB6C-E8EEB4AAE68F) ]
dispinterface IOne
{
?? properties:
?? methods:
?? [id(1), helpstring("メソッドFunOfOne")] void FunOfOne(void);
};
// One のクラス情報(bào)
[ uuid(F329F7D9-477C-4D50-A55B-39A6213215F3) ]
coclass One
{
?? [default] dispinterface IOne;
};
// Two のプライマリ ディスパッチ インターフェイス
??
[ uuid(8BF72FD6-8421-4307-86FB-E4AD4F60161F) ]
dispinterface ITwo
{
?? properties:
?? methods:
?? [id(1), helpstring("メソッドfunOfTwo")] void funOfTwo(void);
};
// Two のクラス情報(bào)
[ uuid(2357C11C-DE0B-4B59-8EA9-B877E98E278B) ]
coclass Two
{
?? [default] dispinterface ITwo;
};
};
如上面所示一個(gè)類型定義文件內(nèi)容,包含兩個(gè)COM類,One和Two,其中One實(shí)現(xiàn)了IOne,Two實(shí)現(xiàn)了ITwo,兩個(gè)類都包含于Fri庫,注意,上在的每個(gè)類,每個(gè)接口和庫都有它的一個(gè)類型GUID.
導(dǎo)入其生成的類型庫,則會看到下面的接口定義
struct __declspec(uuid("7044762e-1832-4997-85ce-f8329f3f1a44"))
/* LIBID */ __Fri;
struct __declspec(uuid("77a9625e-e129-4465-ab6c-e8eeb4aae68f"))
/* dispinterface */ IOne;
struct /* coclass */ One;
struct __declspec(uuid("8bf72fd6-8421-4307-86fb-e4ad4f60161f"))
/* dispinterface */ ITwo;
struct /* coclass */ Two;
//
// Smart pointer typedef declarations
//
_COM_SMARTPTR_TYPEDEF(IOne, __uuidof(IOne));
_COM_SMARTPTR_TYPEDEF(ITwo, __uuidof(ITwo));
...
我只列出了感興趣的一段,包括兩個(gè)智能指針的聲明,可以確定類型庫里有幾個(gè)接口定義,就有幾個(gè)類型指針下面我們這么使用:
IOnePtr ip;
hr = ip.CreateInstance(__uuidof(IOne));
這個(gè)時(shí)候,錯(cuò)誤就發(fā)生了.分析CreateInstance的內(nèi)部,它將根據(jù)你傳入的GUID(上面的__uuidof(IOne))來生成COM對象,換句話說,你應(yīng)該傳入的這個(gè)GUID是一個(gè)COM類的GUID,現(xiàn)在上面的代碼錯(cuò)誤很明顯,它傳入的是接口的IID,雖然GUID和IID是同一種東西,但是用途還是不一樣的(哈哈).另外,智能指針通過模板參數(shù)已經(jīng)明確IOne的IID了.
問題還可以更明白,因你的One 類除了實(shí)現(xiàn)IOne接口還可以實(shí)現(xiàn)其它的更多的接口,或者另外一個(gè)類也實(shí)現(xiàn)了IOne接口,但效果和One類實(shí)現(xiàn)的不一樣,當(dāng)你需要一個(gè)接口指針的時(shí)候,是不是應(yīng)該指明這個(gè)接口是由哪個(gè)類實(shí)現(xiàn)的呢?
[ uuid(F329F7D9-477C-4D50-A55B-39A6213215F3) ]
coclass One
也許你還是不知道該怎么改,如上面改為hr = ip.CreateInstance(__uuidof(One));就行了,雖然One這個(gè)COM類的信息和GUID并不能從typelib導(dǎo)入的類型信息里獲得,但是這樣使用絕對是受微軟保護(hù)的(不會有沒聲明的錯(cuò)誤).
如果這篇文章對你有用,請幫忙頂一下,我也是花了很多時(shí)間才解決問題的.