1

我用一个简单的 COM 对象编写了一个简单的 COM DLL inproc 服务器。COM 对象实现了一个连接点。

我知道如何创建一个派生自 的 ATL 客户端IDispEventImpl,并使用接收器映射来简化此过程。

但是,出于演示的目的,我想创建一个 win32 控制台应用程序,它使用一个调用我的简单 COM 对象的类,然后充当连接点接收器。

我不知道如何提供一个实现IDispatch- 有人可以推荐这方面的文档,因为我找不到任何文档(我有 ATL Internals,但这似乎不能满足我的需要)。

这是我已经上过的课程:

#pragma once
#include <iostream>
using namespace std;

// Because we're implementing a connection points sink (_IPogFarmEvents) 
// in a non-ATL class, we must provide implementations for IUnknown and IDispatch.

class KidWithAPogFarm : public _IPogFarmEvents
{
    private:
        DWORD   m_dwRefCount;
        LONG    m_lNumPogs;

    public:
        KidWithAPogFarm() :
        m_dwRefCount    (0),
        m_lNumPogs  (0)
        {
        }

        ~KidWithAPogFarm()
        {
        }


        // -- IUnknown 
        HRESULT STDMETHODCALLTYPE QueryInterface(REFIID iid, void **ppvObject)
        {
            if (iid == DIID__IPogFarmEvents)
            {
                m_dwRefCount++;
                *ppvObject = (void *)this;
                return S_OK;
            }
            if (iid == IID_IUnknown)
            {
                m_dwRefCount++;
                *ppvObject = (void *)this;
                return S_OK;
            }
            return E_NOINTERFACE;
        }

        ULONG STDMETHODCALLTYPE AddRef()
        {
            m_dwRefCount++;
            return m_dwRefCount;
        }

        ULONG STDMETHODCALLTYPE Release()
        {
            ULONG l;
            l  = m_dwRefCount--;

            if ( 0 == m_dwRefCount)
                delete this;

            return l;
        }


        // -- IDispatch
        STDMETHODIMP GetTypeInfoCount(UINT *pctinfo)
        {       
            return E_NOTIMPL;
        }

        STDMETHODIMP GetTypeInfo( UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo )
        {
            return E_NOTIMPL;
        }

        STDMETHODIMP GetIDsOfNames(const IID &riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId )
        {
            return E_NOTIMPL;
        }
        STDMETHODIMP Invoke(DISPID dispIdMember, const IID &riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr )
        {
            return E_NOT_IMPL;
        }   

        // -- IAntFarmEvents
        STDMETHODIMP OnFarmCreated(LONG lInitialPopulation)
        {
            m_lNumPogs = lInitialPopulation;

            cout << "The kid has a pog farm with " << m_lNumPogs << " pogs " << endl;
            return S_OK;
        }
};
4

4 回答 4

2

既然你已经有了ATL,你可以检查它的来源,看看IDispatchImpl所有这些东西是怎么做的。IDispatch方法是通过从同一模块中的类型库中读取数据来实现的,因为当类型库已经存在时,这是最简单和最可靠的方法。

还值得注意的是,在它上面进行演示是一个相当困难的话题——你需要编写很多代码,而这些代码并不能真正带来任何洞察力。IDispatch如果您实现一个不继承自而是直接继承自的事件接口,IMO 会好得多IUnknown——这将演示事件如何工作,而不会过多地关注IDispatch内部工作。

于 2010-11-04T14:26:22.963 回答
1

我认为最简单的方法是通过CreateStdDispatch

于 2010-11-04T14:28:50.623 回答
1

您可以使用IDispatch 实现。

于 2011-05-11T18:33:30.853 回答
0

这并不完全是您要查找的内容,但FireBreath使用 IDispatchEx 和连接点来提供在 IE 中运行的 ActiveX 控件。因为 FireBreath 是一种允许插件编写一次并在所有主要浏览器中使用的抽象,所以 IDispatch 接口需要手工编写——包括连接点。

代码可能有点混乱,因为有一个模板化的 mixin 类用于向两个不同的 COM 对象类提供 IDispatch 和 ConnectionPoints,但它可能会有所帮助。

您已经接受了答案,但也许它仍然会有所帮助。对不起,我没有早点看到这个问题。

于 2011-05-28T06:14:24.300 回答