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

下載本文檔

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

文檔簡介

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

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

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

4、過typelib,客戶與可連接對象之間的兩種結構,可連接對象的基本結構,可連接對象,如何管理多個出接口每個出接口對應一個連接點對象通過連接點枚舉器管理對于每個出接口,如何管理多個客戶連接通過連接枚舉器管理多個連接,實現可連接對象(源對象)(一),枚舉器內部對象,不需要類廠和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;},實現可連接對象(源對象)(二),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;};,實現可連接對象(源對象)(三),連接點和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; };連接枚舉器 —— 實現IEnumConnections接口允許多個客戶連接每個連接用struct CONNECTDATA來描述,回顧:可連接對象的基本結構,客戶與源對象建立連接過程,客戶請求IConnectionPointContainer接口客戶調用IConnectionPointContainer::FindConnectionPoint找到連

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

14、對象2 通過連接點對象建立連接連接點相當于連接管理器,接收器的實現,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)構造接收器對象?動態(tài)構造vtable?支持部分成員?類型信息的協商通過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ā)函數,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();}},用連接點機制實現回調的討論,比傳統(tǒng)的回調函數功能強大,靈活可以跨進程、跨機器Tightly coupled vs loosely coupled (COM+)要求客戶

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

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

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

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

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

33、功能,ATL實現連接點:最后的工作,在需要激發(fā)事件的地方調用CProxy_提供的輔助函數增加對IProvideClassInfo2接口的支持需要typelib的支持加入基類IProvideClassInfo2Impl在COM MAP中加入:COM_INTERFACE_ENTRY(IProvideClassInfo2)COM_INTERFACE_ENTRY(IProvideClassInfo),ATL實現接收器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中使用出接口,使用瀏覽器控件的事件函數使兩個窗口同步,結構化存儲(structured storage),內容:結構化存儲模型復合文檔永久對象,問題的由來,文件系統(tǒng)的誕生多個應用程序共享同一個存儲設備文件服務功能的抽象進展到結構化存儲

36、多個組件共享同一個文件組件軟件存儲功能的基本要求OLE的需求組件共享句柄方案,如何定位?避免沖突?,文件系統(tǒng)結構,結構化存儲,多個組件程序共享一個復合文件,復合文件,文件內部的文件系統(tǒng)只有兩種對象:存儲對象和流對象實現了部分訪問和增量訪問的功能,流對象,COM庫提供實現,實現了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庫提供實現,實

41、現了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成員函數得到一個子存儲對

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

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

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

50、存儲特性——增量訪問,減少保存和打開文件的時間降低了應用程序對系統(tǒng)資源的要求問題:通過根存儲逐層找到目標對象空間回收,復合文檔,結構化存儲的具體實現底層機制:LockBytes對象把存儲介質描述成一般化的字節(jié)序列復合文檔API函數零內存保存特性,復合文檔模型,,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函數,創(chuàng)建復合文檔的API函數StgCreateDocfile、StgCreateDocfileOnILockBytes打開復合文檔的API函數StgOpenStorage、StgOpenStorageOnILockBytes與內存句柄有關的一組操作函數Crea

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

55、再調用Commit函數即可。,與CLSID的聯系,IStorage::SetClass函數把存儲對象與CLSID聯系起來GetClassFile函數,從文件到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作為數據存儲多個組件協同處理同一個文件->永久對象,永久對象,永久對象實現了IPersistXXX接口的COM對象永久接口:

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

溫馨提示

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

評論

0/150

提交評論