• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 64, 文章 - 11, 評論 - 12, 引用 - 0
            數據加載中……

            ATL連接點開發總結

             

            連接點語義


                     第一種說法:是一種邏輯上的反饋機制,這種機制允許對象暴露其調用一個或者多個指定接口的能力

                     第二種說法:QueryInterface允許客戶從對象中取得一個指向對象實現的接口指針,連接點允許客戶給予對象一個由客戶實現的接口指針.

            在這種情形下:COM對象是源,客戶提供的方法是接收器.

            源必須實現IConnectionPoint

             

            Interface IConnectionPoint

            {

                     HRESULT GetConnectionInterface([out] IID *pIID);

                     HRESULT GetConnectionPointContainer([out] IConnectionPointContainer** ppCPC);

                     HRESULT Advise([in] IUnknnown *punkSing, [Out] DWORD *pdwCookie);

                     HRESULT Unadvise([in]DWORD dwCookie);

                     HREUSLT EnumConnections([out] IEnumConnections** ppEnum);

            }

             

            Interface IConnectionPointContainer

            {

                     HRESULT EnumConnectionPoints([out] IEnumConnectionPoints **ppEnum);

                     HRESULT FindConnectionPoint([in] REFIID riid, [out] IConnectionPoint **ppcP);

            }

             

            客戶的使用方法:

            IUnKnown *pSource;

            ISpeakerEvent *pSink;

            DWORD dwCookie;

            IConnectionPointContainer pcpc;

            Hr = pSource->QueryInterface(&pcpc);

            IConnectionPoint pcp;

            Hr = pcpc->FindConnection(__uuidof(ISpeakerEvent));

            Hr = pcp->Advise(pSink,&dwCookie);

            Hr = pcp->Unadvise(dwCookie);

            便捷的的宏:

            AtlAdivse(psource, pSink, __uuidof(ISpeakerEvent), &dwCookie);

            AtlUnadvise(psource, __uuidof(ISpeakerEvent), dwCookie);

             

             

            建立可連接對象的步聚:

            1:實現IConnectionPointContainer接口

                     Class ATL_NO_VTABLE className:

                               ….

                               Public IConnectionPointContainerImpl<className>

            {…..

            };

            2:QueryInterfaceDIID_IConnectionPointContainer的請求作出響應

                     BEGIN_COM_MAP

            COM_INTERFACE_ENTRY(IConnectionPointContainer)

            END_COM_MAP

             

            3:我們要為每個可連接對象支持的源接口實現IConnectionPoint

                              Class ATL_NO_VTABLE className:

                               ….

                               Public IConnectionPointContainerImpl<className>,

                               Public IConnectionPointImpl<className, &DIID__對外的接口>

            {…..

            };

            4:我們要提供一個連接映射表,也就是一個IID和連接點實現聯系起來的表.

                     BEGIN_CONNECTION_POINT_MAP

                               CONNECTION_POINT_MAP_ENTRY(DIID__對外的接口)

                               ….

                     END_CONNECTION_POINT_MAP()

             

            5:我們必須更新可連接對象在IDL文件中coClass的定義,以便指定每個源接口.每個源接口必須具有屬性,主源接口應具有[default, source]屬性.

                     Coclass 類廠名

                     {……

                               [default,source] dispinterface _對外接口;

                     };

            6:一般來說,我們希望通過輔助方法為所有連接的接收器調用接收器方法.

                     HRESULT Fire_事件(parameter)

                     {

                               依次調用每個接收器的方法

                     }

                    

                     可以使用IDE來生成連接點代理類.這樣我們的源可以從其派生,而不再從IConnectionPointImpl派生.

            7:我們必須在適當的時機調用輔助方法.

             

             

            建立接收事件的對象:

            1:實現事件接收器.可先的方案有從

            IDispEventSimpleImpl<UINT nID, class T, const IID *pdIID = &IID_NULL>

            或者:

            IDispEventImpl< UINT nID, class T, const IID *pdIID = &IID_NULL,

                                        Const GUID*plibid= &GUID_NULL,

                                        DWORD wMajor = 0, WORD wMinor = 0,

                                        Class tihclass = CComTypeInfoHolder>

            派生.

               例如:

               static const int DEFSOURCEID = 1;

               class CEarPolitic;

                     typedef IDispEventImpl< DEFSOURCEID, CEarPolitic, &DIID__ISpeackerEvents

                                        &LIBID_ATLINTERNALSLIB, LIBMAJOR,LIBMINOR> DefSource;

            Class ATL_NO_VTABLE CEarPolitic

                     :public DefSource

            {

                     ….

            }

             

            2:事件接收器映射表

             BEGIN_SINK_MAP(CEarPolitic)

                     SINK_ENTRY_EX(source, DIID, DISPID, EventHandlerFunc)

                     SINK_ENTRY_EX(source, DIID, DISPID, EventHandlerFunc, &info)

             END_SINK_MAP()

             

                     例如:

                     void  __stdcall OnHearPlaintiffWhisper(BSTR bstrText);//sink接口的一個方法.

              

            _ATL_FUNC_INFO OnHearPlaintiffWhisper =

                               {CC_STDCALL, VT_EMPTY, 1, { VT_BSTR}};

            Static const int SOURCEID = 1;

             

            BEGIN_SINK_MAP(CEarPolitic)

                               SINK_ENTRY_EX(SOURCEID, DIID__對外接口, 方法的DISPID, OnHearPlaintiffWhisper)

            END_SINK_MAP()

             

            3:進一步實現這個回調函數.

            4:把事件接收器連接到數據源

             

            posted on 2008-09-30 16:01 Robertxiao 閱讀(2626) 評論(0)  編輯 收藏 引用 所屬分類: RPC/COM/ATL散談

            国产综合免费精品久久久| 久久久久久国产精品无码下载| 国产午夜福利精品久久| 久久久久久亚洲精品不卡| 久久无码精品一区二区三区| 国产精品岛国久久久久| 久久九九免费高清视频| 一个色综合久久| 精品无码人妻久久久久久| 亚洲国产成人精品无码久久久久久综合| 97精品伊人久久大香线蕉| 97久久精品人人澡人人爽| 国产精品久久久久久久久鸭| 久久国产免费| 久久婷婷五月综合色高清| 亚洲午夜久久久久久久久久| 亚洲精品综合久久| 99精品国产在热久久| 国产精品久久久久a影院| 亚洲天堂久久精品| 日韩乱码人妻无码中文字幕久久 | 亚洲精品乱码久久久久66| 国产精品成人99久久久久 | 91精品国产91久久久久久青草| 久久免费视频1| 精品无码人妻久久久久久| 国内精品久久久久久99| 久久亚洲欧美日本精品| 26uuu久久五月天| 久久er99热精品一区二区| 国产精品毛片久久久久久久 | 人妻少妇久久中文字幕| 精品久久久久久久久免费影院| 久久天天躁狠狠躁夜夜2020| 99久久精品免费国产大片| 久久亚洲国产午夜精品理论片| 久久66热人妻偷产精品9| 婷婷综合久久中文字幕蜜桃三电影| 久久人人爽人人人人片av| 久久无码人妻一区二区三区 | 国内精品久久久久影院薰衣草|