com培訓_第1頁
已閱讀1頁,還剩64頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領

文檔簡介

1、COM:可連接對象 & 結(jié)構(gòu)化存儲,潘愛民http://www.icst.pku.edu.cn/CompCourse,內(nèi)容,復習:COM基礎可連接對象結(jié)構(gòu)化存儲,,復習:COM基礎,,COM組件,COM客戶,{IXxx *p;p->…},,聚合模型的關鍵,可連接對象(connectable object),內(nèi)容:可連接對象結(jié)構(gòu)模型實現(xiàn)可連接對象(源對象)客戶-源對象-接收器的協(xié)作過程可連接對象的程

2、序?qū)崿F(xiàn),雙向通信機制 ——客戶與可連接對象的關系,兩個概念,入接口(incoming interface)組件對象實現(xiàn)入接口,客戶通過入接口調(diào)用對象提供的功能客戶和組件都需要知道接口的類型信息出接口(outgoing interface)客戶端提供的COM對象實現(xiàn)出接口組件端的對象通過出接口調(diào)用客戶提供的功能組件提供接口類型信息,客戶實現(xiàn)該接口類似于回調(diào)(callback),但是要復雜和靈活得多,出接口,類型信息

3、由組件一方提供客戶提供出接口的實現(xiàn),實現(xiàn)出接口的COM對象被稱為接收器對象(sink)sink沒有CLSID,也不需要類廠也是一個COM接口,有IID每個成員函數(shù)代表了:事件event通知notification請求request,源對象 or 可連接對象,Connectable object,source普通的COM對象,支持一個或者多個出接口提供出接口的類型信息通過IProvideClassInfo[2]接口通

4、過typelib,客戶與可連接對象之間的兩種結(jié)構(gòu),可連接對象的基本結(jié)構(gòu),可連接對象,如何管理多個出接口每個出接口對應一個連接點對象通過連接點枚舉器管理對于每個出接口,如何管理多個客戶連接通過連接枚舉器管理多個連接,實現(xiàn)可連接對象(源對象)(一),枚舉器內(nèi)部對象,不需要類廠和CLSID其含義就如同指針——智能指針枚舉器接口模板class IEnum : public IUnknown{virtual HRESUL

5、T Next( ULONG celt, ELT_T *rgelt, ULONG *pceltFetched ) = 0;virtual HRESULT Skip( ULONG celt ) = 0;virtual HRESULT Reset( void ) = 0;virtual HRESULT Clone( IEnum**ppenum ) = 0;};,枚舉器的用法,class IStringManage

6、r : public IUnknown { virtual IEnumString* EnumStrings(void) = 0;};void SomeFunc(IStringManager * pStringMan){ String psz; IEnumString * penum; penum=pStringMan->EnumStrings(); while (S_OK == pen

7、um->Next(1, &psz, NULL)) { … //Do something with the string in psz and free it } penum->Release(); return;},實現(xiàn)可連接對象(源對象)(二),IConnectionPointContainer接口class IConnectionPointContainer : p

8、ublic IUnknown { virtual HRESULT EnumConnectionPoints(IEnumConnectionPoints **) = 0; virtual HRESULT FindConnectionPoint(const IID *, IConnectionPoint **) = 0;};IEnumConnectionPoints接口class IEnumConnectionPoints

9、: public IUnknown{virtual HRESULT Next( ULONG cConnections, IConnectionPoint **rgpcn, ULONG *pcFetched) = 0; virtual HRESULT Skip( ULONG cConnections) = 0;virtual HRESULT Reset(void) = 0;virtual HRESULT C

10、lone( IEnumConnectionPoints **ppEnum) = 0;};,實現(xiàn)可連接對象(源對象)(三),連接點和IConnectionPoint接口class IConnectionPoint : public IUnknown { virtual HRESULT GetConnectionInterface( IID *pIID) = 0; virtual HRESULT GetCo

11、nnectionPointContainer( IConnectionPointContainer **ppCPC) = 0; virtual HRESULT Advise( IUnknown *pUnk, DWORD *pdwCookie) = 0; virtual HRESULT Unadvise( DWORD dwCookie) = 0; virtual HRESULT EnumConne

12、ctions(IEnumConnections**ppEnum) = 0; };連接枚舉器 —— 實現(xiàn)IEnumConnections接口允許多個客戶連接每個連接用struct CONNECTDATA來描述,回顧:可連接對象的基本結(jié)構(gòu),客戶與源對象建立連接過程,客戶請求IConnectionPointContainer接口客戶調(diào)用IConnectionPointContainer::FindConnectionPoint找到連

13、接點對象客戶調(diào)用IConnectionPoint::Advise建立與接收器的連接最后,客戶調(diào)用IConnectionPoint::Unadvise取消連接,并釋放連接點對象,客戶方基本結(jié)構(gòu),客戶方實現(xiàn)接收器對象(sink)支持多個與可連接對象之間的連接一般只實現(xiàn)專用的出接口(IUnknown除外)不需要類廠、CLSID與客戶代碼緊密連接起來建立連接1 通過IConnectionPointContainer接口找到連接點

14、對象2 通過連接點對象建立連接連接點相當于連接管理器,接收器的實現(xiàn),class CSomeEventSet : public ISomeEventSet { private: ULONG m_cRef; // Reference count ...... // other private data members public: DWOR

15、D m_dwCookie; // Connection key public: CSomeEventSet (); ~CSomeEventSet(void); //IUnknown members STDMETHODIMP QueryInterface(REFIID, PPVOID); STDMETHODIMP_(DWORD) Ad

16、dRef(void); STDMETHODIMP_(DWORD) Release(void); STDMETHODIMP SomeEventFunction ( ... ); ......};,接收器的用法,ISomeEventSet *gpSomeEventSet;.......// InitializeCSomeEventSet *pSink = new CSomeEven

17、tSet;pSink->QueryInterface(IID_ISomeEventSet, pSomeEventSet ); // Reference count is 1.......// connections the sink object to the connectable object we havehr=pConnectionPoint->Advise(pSomeEventSet , & pS

18、omeEventSet->m_dwCookie);....…// disconnections the sink object from the connectable object we havehr=pConnectionPoint->Unadvise( pSomeEventSet->m_dwCookie);.......// UninitializepSink->Release( ); //

19、Reference count is 0,事件的激發(fā)和處理,BOOL CSourceObject::FireSomeEvent(IConnctionPoint *pConnectionPoint){ IEnumConnections *pEnum; CONNECTDATA connectionData; if (FAILED(pConnectionPoint->EnumConnections

20、(&pEnum))) return FALSE; while (pEnum->Next(1, & connectionData, NULL) == NOERROR) { ISomeEventSet *pSomeEventSet; if (SUCCEEDED(connectionData.pUnk->QueryIn

21、terface(IID_ISomeEventSet, (PPVOID)& pSomeEventSet))) { pSomeEventSet->SomeEventFunction(); // Trigger event or request pSomeEventSet->Release(); } }

22、 pEnum->Release(); return TRUE;},與出接口有關的類型信息,客戶如何知道出接口?運行時刻?編譯時刻?動態(tài)構(gòu)造接收器對象?動態(tài)構(gòu)造vtable?支持部分成員?類型信息的協(xié)商通過IProvideClassInfo[2]能否用標準的接口作為出接口?,用IDispatch接口作為出接口(一),IDispatch接口class IDispatch : public IUnknown{

23、 public: virtual HRESULT GetTypeInfoCount( UINT *pctinfo) = 0; virtual HRESULT GetTypeInfo( UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) = 0; virtual HRESULT GetIDsOfNames(REFIID riid, LPOLEST

24、R *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) = 0; virtual HRESULT Invoke( DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIA

25、NT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) = 0;};,用IDispatch接口作為出接口(二),IDispatch出接口的事件激發(fā)函數(shù),void CMySourceObj::FireMyMethod (short nInt){COleDispatchDriver driver;POSITION pos = m_xMyEventSet.GetStartPo

26、sition();LPDISPATCH pDispatch;while (pos != NULL) {pDispatch = (LPDISPATCH) m_xMyEventSet.GetNextConnection(pos);ASSERT(pDispatch != NULL);driver.AttachDispatch(pDispatch, FALSE);TRYdriver.InvokeHelpe

27、r(eventidMyMethod, DISPATCH_METHOD, VT_EMPTY, NULL,(BYTE *) (VTS_I2), nInt);END_TRYdriver.DetachDispatch();}},用連接點機制實現(xiàn)回調(diào)的討論,比傳統(tǒng)的回調(diào)函數(shù)功能強大,靈活可以跨進程、跨機器Tightly coupled vs loosely coupled (COM+)要求客戶

28、與組件同步?jīng)]有第三方的參與,所以雙方必須保持共識,MFC對連接和事件的支持,用MFC實現(xiàn)源對象,創(chuàng)建工程——支持COM定義出接口——編輯.odl文件利用MFC宏加入連接點聲明以及連接點對象的定義在對象構(gòu)造函數(shù)中調(diào)用EnableConnections();在接口映射表中加入接口IConnectionPointContainer的表項,再加入連接映射表定義連接點類的虛函數(shù)(至少為GetIID)加入事件激發(fā)函數(shù),用MFC在客戶程

29、序中實現(xiàn)接收器,初始化 —— AfxOleInit定義出接口成員類實現(xiàn)出接口成員類創(chuàng)建源對象建立連接和取消連接完成可觸發(fā)事件的動作,用MFC實現(xiàn)的例子,ATL實現(xiàn)可連接對象,在IDL中定義一個用作出接口的automation接口在coclass中加入出接口,含source屬性增加IConnectionPointContainer接口在基類列表中增加IConnectionPointConntainerImpl在CO

30、M MAP中加入COM_INTERFACE_ENTRY(IConnectionPointConntainer),模板類IConnectionPointImpl,CMyClass繼承IConnectionPointImpl一次或多次IConnectionPointImpl實現(xiàn)了獨立的引用計數(shù)用法:在基類列表中增加IConnectionPointImpl加入connection point map,如下BEGIN_CONNEC

31、TION_POINT_MAP(CMyClass)CONNECTION_POINT_ENTRY(DIID__IEventSet)END_CONNECTION_POINT_MAP(),激發(fā)事件輔助函數(shù),手工激發(fā)事件IConnectionPointImpl包含一個m_vec成員,內(nèi)含所有已經(jīng)建立的接收器連接遍歷m_vec數(shù)組,逐一調(diào)用Invoke函數(shù)利用VC IDE提供的源碼產(chǎn)生工具ATL連接點代理生成器,啟動對話框Implem

32、ent Connection Point產(chǎn)生名為CProxy_的模板類例如CProxy_IEventSet,它從IConnectionPointImpl派生對于每一個事件或者請求,都有一個對應的Fire_Xxx成員函數(shù)用模板類代替IConnectionPointImpl基類,Implement Connection Point對話框,創(chuàng)建對象時選擇Connection PointClassView中,在對象類上右鍵點擊選擇此項

33、功能,ATL實現(xiàn)連接點:最后的工作,在需要激發(fā)事件的地方調(diào)用CProxy_提供的輔助函數(shù)增加對IProvideClassInfo2接口的支持需要typelib的支持加入基類IProvideClassInfo2Impl在COM MAP中加入:COM_INTERFACE_ENTRY(IProvideClassInfo2)COM_INTERFACE_ENTRY(IProvideClassInfo),ATL實現(xiàn)接收器sink,I

34、DispEventSimpleImpl輕量,不需要typelib的支持IDispEventImpl需要typelib的支持Event Sink MapBEGIN_SINK_MAP(CMyCLass)SINK_ENTRY_EX(...)// 適合用于non-UI objectSINK_ENTRY(...) // 適合用于UI objectEND_SINK_MAP,ATL:建立sink和source之間的連接,IDi

35、spEventSimpleImpl成員DispEventAdviseDispEventUnadviseAtlAdviseSinkMap建立sink與source缺省源接口的連接,VB中使用出接口,使用瀏覽器控件的事件函數(shù)使兩個窗口同步,結(jié)構(gòu)化存儲(structured storage),內(nèi)容:結(jié)構(gòu)化存儲模型復合文檔永久對象,問題的由來,文件系統(tǒng)的誕生多個應用程序共享同一個存儲設備文件服務功能的抽象進展到結(jié)構(gòu)化存儲

36、多個組件共享同一個文件組件軟件存儲功能的基本要求OLE的需求組件共享句柄方案,如何定位?避免沖突?,文件系統(tǒng)結(jié)構(gòu),結(jié)構(gòu)化存儲,多個組件程序共享一個復合文件,復合文件,文件內(nèi)部的文件系統(tǒng)只有兩種對象:存儲對象和流對象實現(xiàn)了部分訪問和增量訪問的功能,流對象,COM庫提供實現(xiàn),實現(xiàn)了IStream接口class IStream : public IUnknown{public :virtual HRESULT Rea

37、d (void *pv, unsigned long cb, unsigned long *pcbRead) = 0;virtual HRESULT Write (void *pv, unsigned long cb, unsigned long *pcbWritten) = 0;virtual HRESULT Seek (LARGE_INTEGER dlibMove, unsigned long dwOrigin,UL

38、ARGE_INTEGER *plibNewPosition) = 0;virtual HRESULT SetSize (ULARGE_INTEGER libNewSize) = 0;virtual HRESULT CopyTo (LPSTREAM pStm, ULARGE_INTEGER cb,ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten) = 0;virtu

39、al HRESULT Commit (unsigned long dwCommitFlags) = 0;virtual HRESULT Revert ()= 0;virtual HRESULT LockRegion (ULARGE_INTEGER libOffset, ULARGE_INTEGER cb,unsigned long dwLockType) = 0;virtual HRESULT UnlockRegio

40、n (ULARGE_INTEGER libOffset, ULARGE_INTEGER cb,unsigned long dwLockType) = 0;virtual HRESULT Stat (STATSTG *pStatStg, unsigned long grfStatFlag) = 0;virtual HRESULT Clone(LPSTREAM * ppStm) = 0;};,存儲對象,COM庫提供實現(xiàn),實

41、現(xiàn)了IStorage接口class IStorage : public IUnknown{virtual HRESULT CreateStream (const WCHAR * , unsigned long , LPSTREAM * ) = 0;virtual HRESULT OpenStream (const WCHAR * , unsigned long , LPSTREAM * ) = 0;virtual HRE

42、SULT CreateStorage (const WCHAR * , unsigned long ,LPSTORAGE * ) = 0; virtual HRESULT OpenStorage (const WCHAR* , LPSTORAGE *, unsigned long , SNB , unsigned long , LPSTORAGE * ) = 0;virtual HRESULT CopyTo(unsi

43、gned long , IID const *, SNB snbExclude, LPSTORAGE * pStgDest) = 0;virtual HRESULT MoveElementTo(const WCHAR * , LPSTORAGE *,char const * , unsigned long ) = 0;virtual HRESULT Commit (unsigned long ) = 0;virtual H

44、RESULT Revert ()= 0;virtual HRESULT EnumElements (unsigned long , void *,unsigned long , LPENUMSTATSTG * ) = 0;virtual HRESULT DestroyElement (const WCHAR * pwcsName) = 0;virtual HRESULT RenameElement (const WCHAR

45、* pwcsOldName, const WCHAR * pwcsNewName) = 0;virtual HRESULT SetElementTimes(const WCHAR *,FILETIME const *,FILETIME const*,FILETIME const *) = 0;virtual HRESULT SetClass (REFCLSID rclsid) = 0;virtual HRESUL

46、T SetStateBits (unsigned long grfStateBits, unsigned long grfMask) = 0;virtual HRESULT Stat (STATSTG *pStatStg, unsigned long grfStatFlag) = 0;};,客戶如何獲取存儲對象和流對象,如何得到指向根存儲對象的接口指針?CreateStorage和OpenStorage成員函數(shù)得到一個子存儲對

47、象,是唯一的途徑CreateStream和OpenStream成員函數(shù)得到一個流對象,也是唯一的途徑,用結(jié)構(gòu)化存儲設計應用(一),用普通文件組織的文檔結(jié)構(gòu),用結(jié)構(gòu)化存儲設計應用(二),復合文件格式的文檔結(jié)構(gòu),結(jié)構(gòu)化存儲特性——訪問模式,STGM_CREATESTGM_CONVERTSTGM_FAILIFTHERESTGM_DELETEONRELEASESTGM_DIRECTSTGM_TRANSACTEDSTGM_PRIO

48、RITYSTGM_READSTGM_WRITESTGM_READWRITESTGM_SHARE_DENY_READSTGM_SHARE_DENY_WRITESTGM_SHARE_EXCLUSIVESTGM_SHARE_DENY_NONE,結(jié)構(gòu)化存儲特性——事務機制,數(shù)據(jù)一致性和完整性操作:Commit、Revert事務嵌套:以STGM_TRANSACTED標志為基礎事務機制需要消耗較多系統(tǒng)資源Commit參數(shù):S

49、TGC_DEFAULTSTGC_OVERWRITESTGC_ONLYIFCURRENTSTGC_DANGEROUSLYCOMMITMERELYTODISKCACHE,結(jié)構(gòu)化存儲特性——命名規(guī)則,根存儲對象的名字遵守文件系統(tǒng)的命名約定長度不超過32個字符首字符使用大于32的字符,小于32的字符作為首字符有特殊意義不能使用字符“\”、“/”、“:”和“!”名字“.”和“..”被保留名字保留大小寫,但比較操作大小寫無關,結(jié)構(gòu)化

50、存儲特性——增量訪問,減少保存和打開文件的時間降低了應用程序?qū)ο到y(tǒng)資源的要求問題:通過根存儲逐層找到目標對象空間回收,復合文檔,結(jié)構(gòu)化存儲的具體實現(xiàn)底層機制:LockBytes對象把存儲介質(zhì)描述成一般化的字節(jié)序列復合文檔API函數(shù)零內(nèi)存保存特性,復合文檔模型,,root,,,,,,,,,,,,,LockBytes對象,ILockBytes接口class ILockBytes : public IUnknown

51、{public :virtual HRESULT ReadAt (ULARGE_INTEGER , VOID *pv, unsigned long ,unsigned long *) = 0;virtual HRESULT WriteAt (ULARGE_INTEGER , VOID *pv, unsigned long ,unsigned long *) = 0; virtual HRESULT Fl

52、ush ()= 0;virtual HRESULT SetSize (ULARGE_INTEGER cb) = 0;virtual HRESULT LockRegion (ULARGE_INTEGER , ULARGE_INTEGER ,unsigned long ) = 0;virtual HRESULT UnlockRegion (ULARGE_INTEGER , ULARGE_INTEGER ,unsi

53、gned long ) = 0;virtual HRESULT Stat (STATSTG *, unsigned long ) = 0;};,復合文檔API函數(shù),創(chuàng)建復合文檔的API函數(shù)StgCreateDocfile、StgCreateDocfileOnILockBytes打開復合文檔的API函數(shù)StgOpenStorage、StgOpenStorageOnILockBytes與內(nèi)存句柄有關的一組操作函數(shù)Crea

54、teILockBytesOnHGlobal、GetHGlobalFromILockBytesCreateStreamOnHGlobal、GetHGlobalFromStream其他,零內(nèi)存保存特性,意義:資源耗盡之后,保留修改信息資源預留,對于所有的流對象和存儲對象“Save”操作,只要調(diào)用Commit函數(shù)即可“Save As”操作,利用根存儲對象上的IRootStorage接口,調(diào)用SwitchToFile成員函數(shù),

55、再調(diào)用Commit函數(shù)即可。,與CLSID的聯(lián)系,IStorage::SetClass函數(shù)把存儲對象與CLSID聯(lián)系起來GetClassFile函數(shù),從文件到CLSID:復合文件,直接得到根存儲的CLSID非復合文件:(1) 文件擴展名-〉ProgID-〉CLSID(2) HKEY_CLASSES_ROOT\FileType鍵提供了匹配規(guī)則:HKEY_CLASSES_ROOT FileTyp

56、e {} = ,,, = ,,,,復合文檔與COM的關系,復合文檔技術以COM為基礎應用程序在處理復合文檔時把storage或stream直接交給COM組件來處理COM組件接受storage或stream作為數(shù)據(jù)存儲多個組件協(xié)同處理同一個文件->永久對象,永久對象,永久對象實現(xiàn)了IPersistXXX接口的COM對象永久接口:

57、class IPersist : public IUnknownclass IPersistStream : public IPersistclass IPersistStreamInit : public IPersistclass IPersistFile : public IPersistclass IPersistStorage : public Ipersist永久接口的成員函數(shù):GetClassID、IsDirt

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論