?????????在COM中對象要通過class factory的接口(通常是IClassFactory)來創建。在ATL中,class factory也是從CComObjectRootEx派生的COM類,它跟普通的COM類一樣,也通過CComObject或其同伴類將邏輯功能跟生存期管理分離。
?????????在ATL中,class factory類和普通類對象的創建都可以通過被稱為“創建者”的模板類來創建。
?????????
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;
????}
};
?????????“創建者”通過提供一個靜態函數CreateInstance來創建指定類的實例,并且查詢指定的接口。第一個參數在實例化普通對象和class factory對象時會有不同:對于普通對象,通常會傳遞聚合外部對象的IUnknown指針,如果是非聚合創建,可以傳遞NULL(詳見ATL對COM實體身份的支持四);稍后會介紹在class factory創建時第一個參數的用處。
?????????在ATL中,一個class factory對象只能創建一種類型的COM類實例。在class factory的CreateInstance函數中,它不是硬編碼被創建的對象類型,而是通過一個
函數指針成員變量來創建它所管理的對象,而這個函數指針指向的通常就是某個COM類的“創建者”類的CreateInstance靜態函數的地址,此地址是在class factory對象被它自身的“創建者”的CreateInstance函數創建時通過第一個參數傳進來的,class factory重新定義了SetVoid函數,并且保存這個地址:
?????????
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;
};
?????????根據上面的討論,一個COM類的創建者會像這個樣子:
?????????CComCreator<?CComObject<CMyClass> >
?????????而一個class factory的創建者會像這個樣子:
?????????CComCreator< CComCachedObject<CComClassFactory> >。
?????????
?????????ATL在客戶請求創建CMyClass對象時執行的偽代碼如下:
?????????
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 on 2007-04-10 17:36
michael 閱讀(1544)
評論(0) 編輯 收藏 引用 所屬分類:
心得