• <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>
            隨筆-6  評(píng)論-0  文章-16  trackbacks-0
              2007年4月30日

            COM對(duì)象實(shí)現(xiàn)接口的方法是多種多樣的,常用的有多重繼承、聚合、tear-off等。為了最大限度地支持接口查詢的多樣性,ATL使用表驅(qū)動(dòng)的方式來管理接口查詢,以做到代碼封閉和擴(kuò)展靈活,表?xiàng)l目格式為:

            struct ?_ATL_INTMAP_ENTRY
            {
            ????
            const ?IID * ?piid;?
            ????DWORD_PTR?dw;
            ????_ATL_CREATORARGFUNC
            * ?pFunc;?
            };


            piid是接口ID,dw為自定義參數(shù),pFunc是自定義函數(shù)的地址。在QueryInterface時(shí),ATL調(diào)用pFunc,并將dw傳給pFunc,pFunc可以根據(jù)自己的需要解析dw,獲得查詢必需的信息。ATL預(yù)提供了一些函數(shù)來處理上述常用的查詢請(qǐng)求,如_Creator、_Delegate等,它們被放置于CComObjectRootBase中。

            (1)多重繼承
            ???????????ATL并沒有提供支持多重繼承的函數(shù),而是將pFunc設(shè)置為_ATL_SIMPLEMAPENTRY宏,將dw設(shè)為父類子對(duì)象在子類對(duì)象中的偏移量,當(dāng)搜索表時(shí)發(fā)現(xiàn)pFunc為_ATL_SIMPLEMAPENTRY時(shí),便直接使用偏移量相加獲得結(jié)果。

            ????????????if ?(pEntries -> pFunc? == ?_ATL_SIMPLEMAPENTRY)?
            ????????????{
            ????????????????IUnknown * ?pUnk? = ?(IUnknown * )((INT_PTR)pThis + pEntries -> dw);
            ????????????????pUnk
            -> AddRef();
            ????????????????
            * ppvObject? = ?pUnk;
            ????????????????
            return ?S_OK;
            ????????????}

            ?????????接口映射表項(xiàng)宏為:

            ?????????#define ?COM_INTERFACE_ENTRY(x)\
            ?????????????{
            & _ATL_IIDOF(x),?\
            ?????????????offsetofclass(x,?_ComMapClass),?\
            ?????????????_ATL_SIMPLEMAPENTRY},


            (2)聚合
            ???????????ATL使用_Delegate函數(shù)來支持聚合,此時(shí)dw存儲(chǔ)的是內(nèi)部對(duì)象指針成員變量在外部對(duì)象中的偏移量,_Delegate利用此偏移量獲得內(nèi)部對(duì)象指針,然后將查詢調(diào)用委托給它。

            ????static ?HRESULT?WINAPI?_Delegate( void * ?pv,?REFIID?iid,? void ** ?ppvObject,?DWORD_PTR?dw)
            ????{
            ????????HRESULT?hRes?
            = ?E_NOINTERFACE;
            ????????IUnknown
            * ?p? = ? * (IUnknown ** )((DWORD_PTR)pv? + ?dw);
            ????????
            if ?(p? != ?NULL)
            ????????????hRes?
            = ?p -> QueryInterface(iid,?ppvObject);
            ????????
            return ?hRes;
            ????}
            ???
            ?????????接口映射表項(xiàng)宏為:

            ??????#define?COM_INTERFACE_ENTRY_AGGREGATE(iid,?punk)\
            ??????????{
            &iid,\
            ??????????(DWORD_PTR)offsetof(_ComMapClass,?punk),\
            ??????????_Delegate},
            ???
            (3)tear-off
            ???????????ATL使用_Creator函數(shù)來支持tear-off,此時(shí)dw中存儲(chǔ)的是一個(gè)靜態(tài)_ATL_CREATORDATA結(jié)構(gòu)的地址,_Creator利用其中的指向創(chuàng)建函數(shù)的指針創(chuàng)建一個(gè)tear-off對(duì)象。
            ??????????????????
            ?????????static?HRESULT?WINAPI?_Creator(void*?pv,?REFIID?iid,?void**?ppvObject,?DWORD_PTR?dw)
            ?????????{
            ??????????????_ATL_CREATORDATA
            *?pcd?=?(_ATL_CREATORDATA*)dw;
            ??????????????
            return?pcd->pFunc(pv,?iid,?ppvObject);
            ?????????}

            ?????????接口映射表項(xiàng)宏為:
            ????????????
            ??????#define?COM_INTERFACE_ENTRY_TEAR_OFF(iid,?x)\
            ??????????{
            &iid,\
            ??????????(DWORD_PTR)
            &ATL::_CComCreatorData<\
            ??????????????ATL::CComInternalCreator
            <?ATL::CComTearOffObject<?x?>?>\
            ??????????????
            >::data,\
            ??????????_Creator},

            ??????_ComCreatorData的相關(guān)定義如下:
            ??????
            struct?_ATL_CREATORDATA
            {
            ????_ATL_CREATORFUNC
            *?pFunc;
            };

            template?
            <class?Creator>
            class?_CComCreatorData
            {
            public:
            ????
            static?_ATL_CREATORDATA?data;
            };

            template?
            <class?Creator>
            _ATL_CREATORDATA?_CComCreatorData
            <Creator>::data?=?{Creator::CreateInstance};


            ??????我們還可以定義自己的接口查詢輔助函數(shù)和輔助數(shù)據(jù),通過dw和pFunc來完成各種千奇百怪的接口實(shí)現(xiàn)方式,需要的只是想象力。
            posted @ 2007-04-30 11:37 michael 閱讀(695) | 評(píng)論 (0)編輯 收藏
              2007年4月10日
            ?????????在COM中對(duì)象要通過class factory的接口(通常是IClassFactory)來創(chuàng)建。在ATL中,class factory也是從CComObjectRootEx派生的COM類,它跟普通的COM類一樣,也通過CComObject或其同伴類將邏輯功能跟生存期管理分離。
            ?????????在ATL中,class factory類和普通類對(duì)象的創(chuàng)建都可以通過被稱為“創(chuàng)建者”的模板類來創(chuàng)建。
            ?????????
            template?<class?T1>
            class?CComCreator
            {
            public:
            ????
            static?HRESULT?WINAPI?CreateInstance(void*?pv,?REFIID?riid,?LPVOID*?ppv)
            ????{
            ????????HRESULT?hRes?=?E_OUTOFMEMORY;
            ????????T1
            *?p?=?NULL;
            ????????p?
            =?new?T1(pv);
            ????????
            if?(p?!=?NULL)
            ????????{
            ????????????p
            ->SetVoid(pv);
            ????????????p
            ->InternalFinalConstructAddRef();
            ????????????hRes?
            =?p->FinalConstruct();
            ????????????
            p->InternalFinalConstructRelease();
            ????????????
            if?(hRes?==?S_OK)
            ????????????????hRes?
            =?p->QueryInterface(riid,?ppv);
            ????????????
            if?(hRes?!=?S_OK)
            ????????????????delete?p;
            ????????}
            ????????
            return?hRes;
            ????}
            };

            ?????????“創(chuàng)建者”通過提供一個(gè)靜態(tài)函數(shù)CreateInstance來創(chuàng)建指定類的實(shí)例,并且查詢指定的接口。第一個(gè)參數(shù)在實(shí)例化普通對(duì)象和class factory對(duì)象時(shí)會(huì)有不同:對(duì)于普通對(duì)象,通常會(huì)傳遞聚合外部對(duì)象的IUnknown指針,如果是非聚合創(chuàng)建,可以傳遞NULL(詳見ATL對(duì)COM實(shí)體身份的支持四);稍后會(huì)介紹在class factory創(chuàng)建時(shí)第一個(gè)參數(shù)的用處。
            ?????????在ATL中,一個(gè)class factory對(duì)象只能創(chuàng)建一種類型的COM類實(shí)例。在class factory的CreateInstance函數(shù)中,它不是硬編碼被創(chuàng)建的對(duì)象類型,而是通過一個(gè)函數(shù)指針成員變量來創(chuàng)建它所管理的對(duì)象,而這個(gè)函數(shù)指針指向的通常就是某個(gè)COM類的“創(chuàng)建者”類的CreateInstance靜態(tài)函數(shù)的地址,此地址是在class factory對(duì)象被它自身的“創(chuàng)建者”的CreateInstance函數(shù)創(chuàng)建時(shí)通過第一個(gè)參數(shù)傳進(jìn)來的,class factory重新定義了SetVoid函數(shù),并且保存這個(gè)地址:
            ?????????
            class?CComClassFactory?:
            ????
            public?IClassFactory,
            ????
            public?CComObjectRootEx<CComGlobalsThreadModel>
            {
            public:
            ????
            //?IClassFactory
            ????STDMETHOD(CreateInstance)(LPUNKNOWN?pUnkOuter,?REFIID?riid,?void**?ppvObj)
            ????{
            ????????HRESULT?hRes?=?E_POINTER;
            ????????
            if?(ppvObj?!=?NULL)
            ????????{
            ????????????
            *ppvObj?=?NULL;
            ????????????if?((pUnkOuter?!=?NULL)?&&?!InlineIsEqualUnknown(riid))
            ????????????{
            ????????????????hRes?=?CLASS_E_NOAGGREGATION;
            ????????????}
            ????????????
            else
            ????????????????hRes?
            =?m_pfnCreateInstance(pUnkOuter,?riid,?ppvObj);
            ????????}
            ????????
            return?hRes;
            ????}

            ????STDMETHOD(LockServer)(BOOL?fLock)
            ????{
            ????????
            if?(fLock)
            ????????????_pAtlModule
            ->Lock();
            ????????
            else
            ????????????_pAtlModule
            ->Unlock();
            ????????
            return?S_OK;
            ????}
            ?
            ????void?SetVoid(void*?pv)
            ????{
            ????????m_pfnCreateInstance?
            =?(_ATL_CREATORFUNC*)pv;
            ????}

            ????_ATL_CREATORFUNC
            *?m_pfnCreateInstance;
            };

            ?????????根據(jù)上面的討論,一個(gè)COM類的創(chuàng)建者會(huì)像這個(gè)樣子:
            ?????????CComCreator<?CComObject<CMyClass> >
            ?????????而一個(gè)class factory的創(chuàng)建者會(huì)像這個(gè)樣子:
            ?????????CComCreator< CComCachedObject<CComClassFactory> >。
            ?????????
            ?????????ATL在客戶請(qǐng)求創(chuàng)建CMyClass對(duì)象時(shí)執(zhí)行的偽代碼如下:
            ?????????
            typedef?CComCreator<?CComCachedObject<CComClassFactory>?>?CClassFactoryCreator;
            typedef?CComCreator
            <?CComObject<CMyClass>?>?CMyClassCreator;

            IClassFactory
            *?pcf=NULL;
            CClassFactoryCreator::CreateInstance(
            &CMyClassCreator::CreateInstance,__uuidof(IClassFactory),&pcf);
            pcf
            ->CreateInstance(pOuter,__uuidof(IMyInterface),ppvObj);
            posted @ 2007-04-10 17:36 michael 閱讀(1545) | 評(píng)論 (0)編輯 收藏
              2007年4月6日

            ??????有了CComObjectRootBase、CComObjectRootEx和線程模型特征類這些基礎(chǔ)原料,CComObject和它的伙伴們便可以自由的發(fā)揮它們的想象了。

            1.CComObject —— 普通堆對(duì)象??????
            ???下面是精簡的CComObject模板類的定義:??????

            template? < class ?Base >
            class ?CComObject?:? public ?Base
            {
            public :
            ????CComObject(
            void * ? = ?NULL)? throw ()
            ????{
            ????????_pAtlModule
            -> Lock();
            ????}
            ????
            ????virtual ? ~ CComObject()? throw ()
            ????{
            ????????FinalRelease();
            ??????? _pAtlModule -> Unlock();
            ????}
            ????
            ??? STDMETHOD_(ULONG,?AddRef)()?{ return ?InternalAddRef();}
            ????STDMETHOD_(ULONG,?Release)()
            ????{
            ????????ULONG?l?
            = ?InternalRelease();
            ????????
            if ?(l? == ? 0 )
            ????????????delete?
            this ;
            ????????
            return ?l;
            ????}
            ????
            ????STDMETHOD(QueryInterface)(REFIID?iid,? void ? ** ?ppvObject)? throw ()
            ????{
            return ?_InternalQueryInterface(iid,?ppvObject);}
            ????

            ????
            static ?HRESULT?WINAPI?CreateInstance(CComObject < Base >** ?pp)? throw ();
            };


            “普通堆對(duì)象”是使用最為頻繁的類,它有如下特征:
            (1)在堆中分配內(nèi)存 —— 計(jì)數(shù)歸零時(shí)用delete銷毀
            (2)支持獨(dú)立存在的對(duì)象 —— 使用InternalXXX來實(shí)現(xiàn)IUnknown功能
            (3)產(chǎn)生時(shí)鎖定服務(wù)器,銷毀時(shí)解鎖服務(wù)器 —— 適用于大多數(shù)COM對(duì)象

            2. CComAggObject —— 被聚合堆對(duì)象

            template? < class ?contained >
            class ?CComAggObject?:
            ????
            public ?IUnknown,
            ????
            public ?CComObjectRootEx < ?typename?contained::_ThreadModel::ThreadModelNoCS? >
            {
            public :
            ????CComAggObject(
            void * ?pv)?:?m_contained(pv)
            ????{
            ????????_pAtlModule
            -> Lock();
            ????}
            ????
            HRESULT?FinalConstruct()
            ????{
            ????????CComObjectRootEx
            < contained::_ThreadModel::ThreadModelNoCS > ::FinalConstruct();
            ????????
            return ?m_contained.FinalConstruct();
            ????}
            ????
            void ?FinalRelease()
            ????{
            ????????CComObjectRootEx
            < contained::_ThreadModel::ThreadModelNoCS > ::FinalRelease();
            ????????m_contained.FinalRelease();
            ????}
            ???
            ? virtual ? ~ CComAggObject()
            ????{
            ???????
            ?FinalRelease();
            ????????_pAtlModule -> Unlock();
            ????}

            ????STDMETHOD_(ULONG,?AddRef)()?{
            return ?InternalAddRef();}
            ????STDMETHOD_(ULONG,?Release)()
            ????{
            ????????ULONG?l?
            = ?InternalRelease();
            ????????
            if ?(l? == ? 0 )
            ????????????delete?
            this ;
            ????????
            return ?l;
            ????}
            ????STDMETHOD(QueryInterface)(REFIID?iid,?
            void ? ** ?ppvObject)
            ????{
            ????????HRESULT?hRes? = ?S_OK;
            ????????
            if ?(InlineIsEqualUnknown(iid))
            ????????{
            ????????????
            * ppvObject? = ?( void * )(IUnknown * ) this ;
            ????????????AddRef();
            ????????}
            ????????
            else
            ????????????hRes?
            = ?m_contained._InternalQueryInterface(iid,?ppvObject);
            ????????
            return ?hRes;
            ????}
            ???
            ? static ?HRESULT?WINAPI?CreateInstance(LPUNKNOWN?pUnkOuter,?CComAggObject < contained >** ?pp);

            ????CComContainedObject
            < contained > ?m_contained;
            };


            ? CComAggObject有如下特征:
            (1)在堆中分配內(nèi)存
            (2)支持被聚合聚合的對(duì)象
            ???????????實(shí)現(xiàn)了兩份IUnknown:CComAggObject實(shí)現(xiàn)了內(nèi)部聚合類真正的IUnknown,它管理對(duì)象的生存期,并且完成接口查詢(通過contained對(duì)象的_InternalQueryInterface);contained對(duì)象實(shí)現(xiàn)了具有轉(zhuǎn)發(fā)功能的IUnknown,它將所有的調(diào)用轉(zhuǎn)發(fā)給CComAggObject在構(gòu)造函數(shù)中收到的外部IUnknown指針。
            (3)產(chǎn)生時(shí)鎖定服務(wù)器,銷毀時(shí)解鎖服務(wù)器

            3. CComTearOffObject —— tear-off 對(duì)象

            template? < class ?Base >
            class ?CComTearOffObject?:? public ?Base
            {
            public :
            ????CComTearOffObject(
            void * ?pv)
            ????{
            ????????m_pOwner? = ?reinterpret_cast < Base::_OwnerClass *> (pv);
            ????????m_pOwner
            -> AddRef();
            ????}
            ???? ~ CComTearOffObject()
            ????{
            ????????FinalRelease();
            ????????m_pOwner -> Release();
            ????}

            ????STDMETHOD_(ULONG,?AddRef)()?
            throw ()?{ return ?InternalAddRef();}
            ????STDMETHOD_(ULONG,?Release)()?
            throw ()
            ????{
            ????????ULONG?l?
            = ?InternalRelease();
            ????????
            if ?(l? == ? 0 )
            ????????????delete?
            this ;
            ????????
            return ?l;
            ????}
            ????STDMETHOD(QueryInterface)(REFIID?iid,?
            void ? ** ?ppvObject)? throw ()
            ????{
            ????????
            return ?m_pOwner -> QueryInterface(iid,?ppvObject);
            ????}
            };

            ???tear-off對(duì)象有如下特征:
            (1)在堆中分配內(nèi)存
            (2)支持在ATL組件內(nèi)部使用的tear-off技術(shù)
            (3)生成時(shí)鎖定父對(duì)象,銷毀時(shí)解鎖父對(duì)象
            (4)生命周期由自己管理,接口查詢委托父對(duì)象

            4. CComObjectStack —— 棧對(duì)象
            ???
            template?<class?Base>
            class?CComObjectStackEx?:?public?Base
            {
            public:
            ????CComObjectStackEx(
            void*?=?NULL)?
            ????{?
            ????????m_hResFinalConstruct?=?FinalConstruct();?
            ????}
            ????
            virtual?~CComObjectStackEx()
            ????{
            ???????
            FinalRelease();
            ??? }

            ????STDMETHOD_(ULONG,?AddRef)()
            ????{
            ??????? return?0;
            ????}

            ????STDMETHOD_(ULONG,?Release)()
            ????{
            ????????return?0;
            ????}

            ????STDMETHOD(QueryInterface)(REFIID?iid,?
            void?**?ppvObject)
            ????{
            ????????
            return?_InternalQueryInterface(iid,?ppvObject);
            ????}

            ????HRESULT?m_hResFinalConstruct;
            };

            ???棧對(duì)象有如下特征:
            (1)內(nèi)存分配于棧上 —— 不需要對(duì)象考慮釋放問題
            (2)無引用計(jì)數(shù)功能
            (3)構(gòu)造時(shí)初始化 —— 需要在構(gòu)造函數(shù)調(diào)用后檢查m_hResFinalConstruct,以判斷對(duì)象是否構(gòu)造成功
            (4)不鎖定服務(wù)器

            5. CComObjectGlobal —— 全局對(duì)象
            template?<class?Base>
            class?CComObjectGlobal?:?public?Base
            {
            public:
            ????CComObjectGlobal(
            void*?=?NULL)
            ????{
            ???????
            m_hResFinalConstruct?=?FinalConstruct();
            ????}
            ????
            virtual?~CComObjectGlobal()
            ????{
            ?????????????FinalRelease();
            ??
            ??}
            ????STDMETHOD_(ULONG,?AddRef)()?
            throw()
            ????{
            ????????
            return?_pAtlModule->Lock();
            ????}
            ????STDMETHOD_(ULONG,?Release)()?
            throw()
            ????{
            ????????
            return?_pAtlModule->Unlock();
            ????}
            ????STDMETHOD(QueryInterface)(REFIID?iid,?
            void?**?ppvObject)?throw()
            ????{
            ????????
            return?_InternalQueryInterface(iid,?ppvObject);
            ????}
            ????HRESULT?m_hResFinalConstruct;
            };
            ???
            ???全局對(duì)象有如下特征:
            (1)在全局?jǐn)?shù)據(jù)區(qū)分配內(nèi)存
            (2)初始化時(shí)不鎖定服務(wù)器,引用計(jì)數(shù)變化時(shí)才鎖定或者解鎖服務(wù)器,這使得全局對(duì)象可以被用于全局類廠對(duì)象,以解決服務(wù)器和類廠之間的死鎖問題
            (3)構(gòu)造方式同棧對(duì)象

            ???還有一些其它種類的生存期管理類,如CComObjectNoLock、CComObjectCached、CComPolyObject等等,它們也都有自己獨(dú)到的用處,我們也可以根據(jù)自己的需要編寫自定義的類。總之CComObjectRootBase、CComObjectRootEx和線程模型特征類就像是積木一樣,我們可以任意的把它們組合成想要的形狀。
            posted @ 2007-04-06 15:33 michael 閱讀(3689) | 評(píng)論 (0)編輯 收藏
            1.CComSingleThreadModel和CComMultiThreadModel特征類:
            class?CComSingleThreadModel
            {
            public:
            ????
            static?ULONG?WINAPI?Increment(LPLONG?p)?throw()?{return?++(*p);}
            ????
            static?ULONG?WINAPI?Decrement(LPLONG?p)?throw()?{return?--(*p);}
            ????typedef?CComFakeCriticalSection?AutoCriticalSection;
            ????typedef?CComFakeCriticalSection?CriticalSection;
            ????typedef?CComSingleThreadModel?ThreadModelNoCS;
            };
            class?CComMultiThreadModel
            {
            public:
            ????
            static?ULONG?WINAPI?Increment(LPLONG?p)?throw()?{return?InterlockedIncrement(p);}
            ????
            static?ULONG?WINAPI?Decrement(LPLONG?p)?throw()?{return?InterlockedDecrement(p);}
            ????typedef?CComAutoCriticalSection?AutoCriticalSection;
            ????typedef?CComCriticalSection?CriticalSection;
            ????typedef?CComMultiThreadModelNoCS?ThreadModelNoCS;
            };
            其中的CFakeCriticalSection定義如下:
            class?CComFakeCriticalSection
            {
            public:
            ????HRESULT?Lock()?
            throw()?{?return?S_OK;?}
            ????HRESULT?Unlock()?
            throw()?{?return?S_OK;?}
            ????HRESULT?Init()?
            throw()?{?return?S_OK;?}
            ????HRESULT?Term()?
            throw()?{?return?S_OK;?}
            };

            這兩個(gè)類提供了針對(duì)特定套間對(duì)象的同步控制支持。
            CComSingleThreadModel適用于單線程套間對(duì)象,它不采用任何線程同步機(jī)制;CComMultiThreadModel適用于多線程套間對(duì)象,它使用win32的同步機(jī)制來實(shí)現(xiàn)。

            2.CComObjectRootEx模板類
            template?<class?ThreadModel>
            class?CComObjectRootEx?:?public?CComObjectRootBase
            {
            public:
            ????typedef?ThreadModel?_ThreadModel;
            ????typedef?typename?_ThreadModel::AutoCriticalSection?_CritSec;
            ????typedef?CComObjectLockT
            <_ThreadModel>?ObjectLock;

            ????ULONG?InternalAddRef()
            ????{
            ???????
            ?return?_ThreadModel::Increment(&m_dwRef);
            ????}
            ????ULONG?InternalRelease()
            ????{
            ????????return?_ThreadModel::Decrement(&m_dwRef);
            ????}

            ????
            void?Lock()?{m_critsec.Lock();}
            ????
            void?Unlock()?{m_critsec.Unlock();}

            private:
            ????_CritSec?m_critsec;
            };

            它使用傳入的線程模型特征類來實(shí)現(xiàn)對(duì)象獨(dú)立存在時(shí)所需要的引用計(jì)數(shù)支持,同時(shí)提供了一個(gè)方便對(duì)象開發(fā)使用的同步鎖,我們可以使用這個(gè)鎖來完成對(duì)自定義數(shù)據(jù)訪問的保護(hù)。
            posted @ 2007-04-06 14:39 michael 閱讀(1614) | 評(píng)論 (0)編輯 收藏

            下面是CComObjectRootBase的關(guān)鍵代碼:

            class ?CComObjectRootBase
            {
            public :
            ?????// 構(gòu)造支持
            ?????//
            ??
            ???HRESULT?FinalConstruct();
            ???? void?FinalRelease();?

            ?????//??接口查詢支持
            ?????//
            ?????static?HRESULT?WINAPI?InternalQueryInterface(void*?pThis,
            ????????
            const?_ATL_INTMAP_ENTRY*?pEntries,?REFIID?iid,?void**?ppvObject);
            ???
            ????static?HRESULT?WINAPI?_Break(void*,?REFIID?iid,?void**??,?DWORD_PTR?);
            ????static?HRESULT?WINAPI?_NoInterface(void*??,?REFIID?,?void**??,?DWORD_PTR?);
            ????static?HRESULT?WINAPI?_Creator(void*?pv,?REFIID?iid,?void**ppvObject,?DWORD_PTR?dw);
            ????static?HRESULT?WINAPI?_Delegate(void*?pv,?REFIID?iid,?void**?ppvObject,?DWORD_PTR?dw);
            ???
            ?static?HRESULT?WINAPI?_Chain(void*?pv,?REFIID?iid,?void**?ppvObject,?DWORD_PTR?dw);
            ???
            static?HRESULT?WINAPI?_ChainAttr(void*?pv,?REFIID?iid,?void**?ppvObject,?DWORD_PTR?dw);
            ???
            ?static?HRESULT?WINAPI?_Cache(void*?pv,?REFIID?iid,?void**?ppvObject,?DWORD_PTR?dw);

            ????//? 聚合支持
            ??? //
            ?
            ???ULONG?OuterAddRef()
            ????{
            ????????
            return?m_pOuterUnknown->AddRef();
            ????}
            ????ULONG?OuterRelease()
            ????{
            ????????
            return?m_pOuterUnknown->Release();
            ????}
            ????HRESULT?OuterQueryInterface(REFIID?iid,?
            void?**?ppvObject)
            ????{
            ????????
            return?m_pOuterUnknown->QueryInterface(iid,?ppvObject);
            ????}
            ????
            ??? //?實(shí)體身份支持
            ??? //
            ????union
            ????{
            ????????
            long?m_dwRef;
            ???????? IUnknown
            *?m_pOuterUnknown;
            ????};
            };

            CComObjectRootBase主要功能分為:
            (1)構(gòu)造支持:
            ????????????即FinalConstruct和FinalRelease,它們用于安全的構(gòu)造和銷毀對(duì)象。
            (2)接口查詢支持
            ????????????通過接口映射表來完成接口查詢功能,詳細(xì)內(nèi)容見后面的文章。
            (3)聚合狀態(tài)下引用計(jì)數(shù)操作的支持
            ????????????提供了將IUnknown請(qǐng)求委托給外部組件的支持。當(dāng)對(duì)象獨(dú)立存在時(shí),對(duì)引用計(jì)數(shù)操作的支持被放置于CComObjectRootEx中,因?yàn)樯婕暗搅颂组g和訪問同步。
            (4)實(shí)體身份支持
            ????????????使得對(duì)象可以獨(dú)立存在或者被聚合,因?yàn)檫\(yùn)行時(shí)刻對(duì)象只能以一種方式存在,所以使用了union。
            posted @ 2007-04-06 13:51 michael 閱讀(1515) | 評(píng)論 (0)編輯 收藏
            ATLComClasses.JPG
            ????????????一個(gè)COM對(duì)象必須實(shí)現(xiàn)IUnknown接口所規(guī)定的功能,即生命期管理和運(yùn)行時(shí)類型轉(zhuǎn)換。
            (1)基本IUnknow功能
            ??????????? 對(duì)象的生存方式可以是堆存儲(chǔ)、棧存儲(chǔ)以及全局存儲(chǔ),同時(shí)COM還提供了一種被聚合的生存方式。生存方式不同,AddRef、Release和QueryInterface的實(shí)現(xiàn)方式就不同。比如:對(duì)于堆方式,Release要在引用計(jì)數(shù)歸零時(shí)負(fù)責(zé)釋放堆空間,而棧以及全局方式則不必;對(duì)于聚合方式,AddRef、Release以及大部分的QueryInterface調(diào)用要委托給外部對(duì)象。
            ??????????? 為了達(dá)到對(duì)象的邏輯功能與其生存方式的分離,COM把對(duì)IUnknown基本功能的實(shí)現(xiàn)推遲到類層次結(jié)構(gòu)的最底層,即CComObject(或它的其他伙伴)處。
            ??????????? CComObjectRootBase為CComObject的實(shí)現(xiàn)提供原材料的支持,如提供引用計(jì)數(shù)變量、基本的接口查詢機(jī)制等,它是一個(gè)“功能集中類”。
            (2)套間安全支持
            ??????????? 根據(jù)對(duì)象所處的套間類型,對(duì)象功能的實(shí)現(xiàn)需要不同的同步支持。對(duì)于單線程套間對(duì)象,由COM庫來保證訪問的同步;而對(duì)于多線程套間對(duì)象,需要自己來同步調(diào)用。為了使套間安全支持與邏輯功能分離,ATL提供了一組提供線程安全操作支持的特征類(traits類)—— CComSingleThreadModel及其伙伴。它們提供了對(duì)一個(gè)LONG變量“恰好線程安全”的增減操作,以及跟套間類型匹配的CriticalSection功能類。
            ??????????? ATL同時(shí)提供了CComObjectRootEx模板類,它以某種ThreadModel特征類作為參數(shù),提供對(duì)引用計(jì)數(shù)“恰好線程安全”的操作,并且提供了一個(gè)用于對(duì)象同步的鎖。
            ????????????PS:“恰好線程安全”的操作的意思是:對(duì)于需要線程同步的對(duì)象,提供win32 CriticalSection支持;對(duì)于不需要的對(duì)象,則什么也不做。
            posted @ 2007-04-06 11:46 michael 閱讀(1384) | 評(píng)論 (0)編輯 收藏
            僅列出標(biāo)題  
            99久久精品免费看国产一区二区三区 | 激情五月综合综合久久69| 亚洲精品tv久久久久| 国内精品久久久久久不卡影院| 狠狠色丁香久久综合五月| 国内精品人妻无码久久久影院导航| 久久亚洲中文字幕精品一区四 | 国产精品久久精品| 久久精品毛片免费观看| 人妻少妇久久中文字幕一区二区| 伊人久久精品无码av一区| 婷婷国产天堂久久综合五月| 韩国三级中文字幕hd久久精品 | 中文字幕热久久久久久久| 精品久久久久久久国产潘金莲| 亚洲伊人久久综合中文成人网| 免费无码国产欧美久久18| 久久久久精品国产亚洲AV无码| 77777亚洲午夜久久多喷| 无码人妻久久一区二区三区免费丨| 亚洲精品乱码久久久久久按摩| 久久久国产精品亚洲一区| 精品熟女少妇a∨免费久久| 成人国内精品久久久久影院| 国产精品综合久久第一页| 日韩久久久久中文字幕人妻| 国产美女亚洲精品久久久综合| 久久精品亚洲日本波多野结衣| 99久久精品免费看国产一区二区三区| 很黄很污的网站久久mimi色| 久久久久亚洲AV无码专区首JN | 成人久久精品一区二区三区| 久久国产精品免费一区二区三区| 伊人久久大香线蕉无码麻豆| 亚洲精品乱码久久久久久| 亚洲国产精品久久久久网站| 国产精品久久久久免费a∨| 精品久久久久久综合日本| 婷婷久久综合九色综合九七| 久久人爽人人爽人人片AV| 久久综合给合综合久久|