• <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 閱讀(2638) 評論(0)  編輯 收藏 引用 所屬分類: RPC/COM/ATL散談

            欧美亚洲另类久久综合婷婷| 久久国产高清字幕中文| 2021国产精品久久精品| 亚洲午夜无码久久久久| 99久久精品国产毛片| 久久热这里只有精品在线观看| 国产成人久久精品一区二区三区 | 成人综合伊人五月婷久久| 亚洲国产精品久久久久网站| 久久精品国产久精国产果冻传媒 | 亚洲综合婷婷久久| 国内精品久久久久影院亚洲| 99久久99这里只有免费费精品| 人人狠狠综合88综合久久| 久久久久人妻一区二区三区vr| 久久久免费观成人影院| 亚洲狠狠久久综合一区77777| 久久伊人五月丁香狠狠色| 国产L精品国产亚洲区久久| 久久久久久久久久久久中文字幕| 中文字幕久久亚洲一区| 久久久久亚洲爆乳少妇无| 国产精品免费看久久久| 久久久久99精品成人片欧美| 久久无码专区国产精品发布 | 久久久久97国产精华液好用吗| 国产99久久精品一区二区| 青草国产精品久久久久久| 综合人妻久久一区二区精品| 99久久夜色精品国产网站| 国内精品伊人久久久久妇| 狠狠色丁香久久婷婷综合图片| 久久久久一本毛久久久| 看全色黄大色大片免费久久久| 久久久久99精品成人片| 性高朝久久久久久久久久| 色婷婷噜噜久久国产精品12p| 国产精品99久久久久久宅男小说| 伊人久久国产免费观看视频 | 久久久久这里只有精品| 久久婷婷是五月综合色狠狠|