6

イベントを含むC#COMオブジェクトを作成することができました。以下のコードを見つけてください、

    [Guid("1212674-38748-45434")]
    public interface ICalculator
    {
        int Add(int Num1, int Num2);
    }

    [InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    [Guid("3453674234-84444-84784")]
    public interface ICalculatorEvents
    {
        [DispId(1)]
        void Completed(int Result);
    }

    [ClassInterface(ClassInterfaceType.None)]
    [ComSourceInterfaces(typeof(ICalculatorEvents))]
    [Guid("87457845-945u48-4954")]
    public class Calculator : ICalculator
    {
        public delegate void CompletedDelegate(int result);
        public event CompletedDelegate Completed;
        public Add(int Num1, int Num2)
        {
            int Result = Num1 + Num2;
            if(Completed != null)
                Completed(Result);
        }
    }

このCOMオブジェクトをC++コンソールアプリケーションにインポートし、「Add()」メソッドを呼び出すことができます。C++アプリケーションで「Completed」イベントを処理する方法がわかりません。これについてアドバイスしてもらえますか?このイベントが発生するたびに、結果の値をコンソールに表示したいと思っています。

以下のC++アプリケーションのコードを見つけてください。イベント「完了」はここでは処理されません。これは無限ループに入ります。

    #import "Calculator.tlb"
    using namespace Calculator;
    int Flag = 0;
    class HandleEvent : public ICalculatorEvent
    {
        public:
            HandleEvent(void);
            ~HandleEvent(void);
            HRESULT __stdcall QueryInterface(const IID &, void **);
            ULONG __stdcall AddRef(void) { return 1; }
            ULONG __stdcall Release(void) { return 1; }
            HRESULT __stdcall Completed(int Result);
    };

    HandleEvent::HandleEvent(void)
    {
    }

    HRESULT HandleEvent::Completed(int Result)
    {
        printf("Addition Completed, Result: %d", Result);
        Flag = 1;
    }

    HRESULT HandleEvent::QueryInterface(const IID & iid,void ** pp)
    {
        if (iid == __uuidof(ICalculatorEvent) || iid == __uuidof(IUnknown))
        {
            *pp = this;
            AddRef();
            return S_OK;
        }
        return E_NOINTERFACE;
    }

    int _tmain(int argc, _TCHAR* argv[])
    {
        CoInitialize(NULL);
        Flag = 0;
        ICalculatorPtr pCalc(__uuidof(Calculator));
        pCalc->Add(5, 6);

        do
        {
        }while(Flag == 0);

        CoUninitialize ();
        return 0;
    }

前もって感謝します。

4

2 に答える 2

0

デリゲートを使用する場合は、インターフェイスを宣言する必要はありません。_tmain() 関数を次のように変更します。

int _tmain(int argc, _TCHAR* argv[])
{
    CoInitialize(NULL);
    Flag = 0;

    EventHandler evh ;
    ICalculatorPtr pCalc(__uuidof(Calculator));
    pCalc->Completed = &evh.Completed() ;
    pCalc->Add(5, 6);

    do
    {
    }while(Flag == 0);

    CoUninitialize ();
    return 0;
}

インターフェイスを使用する場合は、これを試してください。

 [ClassInterface(ClassInterfaceType.None)]
 [ComSourceInterfaces(typeof(ICalculatorEvents))]
 [Guid("87457845-945u48-4954")]
 public class Calculator : ICalculator
 {
     public ICalculatorEvents callbackObject ;

     public Add(int Num1, int Num2)
     {
         int Result = Num1 + Num2;
         if(callbackObject != null)
             callbackObject.Completed(Result);
     }
 }

_tmain() メソッドをこれに変更します。

int _tmain(int argc, _TCHAR* argv[])
{
    CoInitialize(NULL);
    Flag = 0;

    EventHandler evh ;
    ICalculatorPtr pCalc(__uuidof(Calculator));
    pCalc->callbackObject = &evh ;
    pCalc->Add(5, 6);

    do
    {
    }while(Flag == 0);

    CoUninitialize ();
    return 0;
}
于 2012-06-07T00:13:02.277 に答える
0

C++ クライアントでの COM 初期化は、次を使用して行う必要があることがわかりました。

    CoInitializeEx(NULL, COINIT_MULTITHREADED);

C# (.NET) COM サーバーからの非同期イベント処理の場合、それ以外の場合、C++ クライアントは CoUninitialize() 呼び出しの後にのみイベントを受け取ります。

イベント処理クラス:

    class EventWrapper : public IDispEventSimpleImpl<1, EventWrapper, &DIID_RumCardCOMEvents >
    {
    public:
        // now you need to declare a sink map - a map of methods handling the events
        BEGIN_SINK_MAP(EventWrapper)
            SINK_ENTRY_INFO(1, DIID_RumCardCOMEvents, 0x1, isCardInserted, &cardInserted)
            SINK_ENTRY_INFO(1, DIID_RumCardCOMEvents, 0x2, isCardRemoved, &cardRemoved)
            // event interface id (can be more than 1)---+      |      |                   |
            // must match dispid of your event -----------------+      |                   |
            // method which handles the event  ------------------------+                   |
            // type information for event, see below --------------------------------------+
        END_SINK_MAP()

    // declare the type info object. You will need one for each method with different signature.
        // it will be defined in the .cpp file, as it is a static member
        static _ATL_FUNC_INFO cardInserted;  // 'placeholder' object to carry event information (see below)
        static _ATL_FUNC_INFO cardRemoved;  // 'placeholder' object to carry event information (see below)

        // method which handles the event
        STDMETHOD (isCardInserted)(unsigned char type)
        { 
            // usually it is defined it in the .cpp file
            cout << "isCardInserted: " << (int)type << endl;
            return 0;
        }

        STDMETHOD (isCardRemoved)()
        { 
            // usually it is defined it in the .cpp file
            cout << "isCardRemoved" << endl;
            return 0;
        }
    };

主要:

    int main()
    {
        CoInitializeEx(NULL, COINIT_MULTITHREADED);
        try
        {
            EventWrapper ev;
            ev.DispEventAdvise(/*COM interface*/);
            // receiving events
            ev.DispEventUnadvise(/*COM interface*/);
        }
        catch (_com_error& e)
        {
            cout << "Exception: " << e.ErrorMessage() << endl;
        }

        CoUninitialize();
        return 0;
    }
于 2013-04-19T08:18:57.170 に答える