複数のデュアル インターフェイスを持つ COM オブジェクトを作成しました。以前のバージョンのコンパイラでは機能しましたが、現在のバージョンでは機能しませんでした。
私の質問: COM の仕様では、これが機能するはずである (したがって、コンパイラのバグを報告する必要がある) と書かれていますか、それとも機能するように意図されていませんか? このページは、おそらく機能しないことを示唆しています。
RIDL ファイル:
[
uuid(0A6CC6CE-623E-4455-8B9B-65178FB7585A),
version(1.0),
helpstring("Library to illustrate failure of Dispatch interface")
]
library DaxFail
{
importlib("stdole2.tlb");
interface IFoo;
coclass DaxFailClass;
interface IBar;
[
uuid(2CD15FFC-0C09-4A29-BD57-99BBC53AE01F),
helpstring("Dispatch interface for DaxFailClass Object"),
dual,
oleautomation
]
interface IFoo: IDispatch
{
[id(0x000000C9)]
HRESULT _stdcall foo_method(void);
};
[
uuid(AECB5DF3-EDE3-441A-93E6-220CB271AD43),
dual,
oleautomation
]
interface IBar: IDispatch
{
[id(0x000000C9)]
HRESULT _stdcall bar_method(void);
};
[
uuid(9DCD1024-6E1A-435E-82F9-FD4FE863D710),
helpstring("DaxFailClass Object")
]
coclass DaxFailClass
{
[default] interface IFoo;
interface IBar;
};
};
それにアクセスするためのコード (これは疑似コードです。「実際の」コードには、HRESULT
s を表示するための追加のステートメントがあります):
const GUID CLSID_DaxFailClass = {0x9DCD1024, 0x6E1A, 0x435E,{ 0x82, 0xF9, 0xFD,0x4F, 0xE8, 0x63,0xD7, 0x10} };
const GUID IID_IFoo = {0x2CD15FFC, 0x0C09, 0x4A29,{ 0xBD, 0x57, 0x99,0xBB, 0xC5, 0x3A,0xE0, 0x1F} };
const GUID IID_IBar = {0xAECB5DF3, 0xEDE3, 0x441A,{ 0x93, 0xE6, 0x22,0x0C, 0xB2, 0x71,0xAD, 0x43} };
int _tmain()
{
IDispatch *intf, *ibar;
HRESULT hr;
DISPID disp_id;
wchar_t *name;
CoInitialize(NULL);
hr = CoCreateInstance(CLSID_DaxFailClass, 0, CLSCTX_ALL, IID_IDispatch, (void **)&intf);
// This returns 0 , and doing Invoke with disp_id executes foo_method
name = L"foo_method";
intf->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_SYSTEM_DEFAULT, &disp_id );
// returns 0
hr = intf->QueryInterface(IID_IBar, (void **)&ibar);
// this returns 0x80020006
name = L"bar_method";
hr = ibar->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_SYSTEM_DEFAULT, &disp_id );
// This returns 0 , and doing Invoke with disp_id executes foo_method
name = L"foo_method";
hr = ibar->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_SYSTEM_DEFAULT, &disp_id );
CoUninitialize();
getchar();
}
したがって、問題は、が とibar
まったく同じように動作することintf
です。ibar
それを呼び出した可能性がありますが、何がfoo_method
何であるかを知らないようですbar_method
。
GetIDsOfNames
2 番目の呼び出しで が呼び出され0
、次に Invoke で が呼び出さbar_method
れ、3 番目の GetIDsOfNames で が呼び出されることを期待していました0x80020006
。
コンパイラに関する追加情報 (ただし、明確にするために、私の質問は、COM 仕様が動作するはずであると述べているかどうかです): BDS 2006 で動作し、C++Builder XE5 では動作しません。COM を実装する XE5 のコードを調べてみると、オブジェクトが最初に作成されたときにojbect ファクトリが using を埋めITypeInfo *
ますが、の実装はすべての結果に対して同じものを使用するだけで、 new で再度呼び出すことはありません。これはの実装で に渡されます。GetTypeInfoOfGUID(CLSID_....)
QueryInterface
ITypeInfo
GetTypeInfoOfGUID
IID
ITypeInfo
DispGetIDsOfNames
IDispatch::GetIDsOfNames