1

複数のデュアル インターフェイスを持つ 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;
  };

};

それにアクセスするためのコード (これは疑似コードです。「実際の」コードには、HRESULTs を表示するための追加のステートメントがあります):

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

GetIDsOfNames2 番目の呼び出しで が呼び出され0、次に Invoke で が呼び出さbar_methodれ、3 番目の GetIDsOfNames で が呼び出されることを期待していました0x80020006

コンパイラに関する追加情報 (ただし、明確にするために、私の質問は、COM 仕様が動作するはずであると述べているかどうかです): BDS 2006 で動作し、C++Builder XE5 では動作しません。COM を実装する XE5 のコードを調べてみると、オブジェクトが最初に作成されたときにojbect ファクトリが using を埋めITypeInfo *ますが、の実装はすべての結果に対して同じものを使用するだけで、 new で再度呼び出すことはありません。これはの実装で に渡されます。GetTypeInfoOfGUID(CLSID_....)QueryInterfaceITypeInfoGetTypeInfoOfGUIDIIDITypeInfoDispGetIDsOfNamesIDispatch::GetIDsOfNames

4

1 に答える 1