亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

COM 組件設計與應用(一)起源及復合文件

2019-11-17 05:20:45
字體:
來源:轉載
供稿:網友
一、前言

  公元一九九五年某個夜黑風高的晚上,我的一位老師跟我說:“小楊呀,以后寫程序就和搭積木一樣啦。你趕緊學習一些OLE的技術吧......”,當時我心里就尋思 :“開什么玩笑?搭積木方式寫程序?再過100年吧......”,但作為一名聽話的好學生,我開始在書店里“踅摸”(注1)有關OLE的書籍(注2)。功夫不負有心人,終于買到了我的第一本COM書《OLE2 高級編程技術》,這本800多頁的大布頭花費了我1/5的月工資呀......于是開始日夜耕讀.....
功夫不負有心人,我堅持讀完了全部著作,感想是:這本書,在說什么吶?
功夫不負有心人,我又讀完了一遍大布頭,感想是:咳~~~,沒懂!
功夫不負有心人,我再,我再,我再讀 ... 感想是:哦~~~,讀懂了一點點啦,哈哈哈。
...... ......
功夫不負有心人,我終于,我終于懂了。
800頁的書對現在的我來說,其實也就10幾頁有用。到這時候才體會出什么叫“書越讀越薄”的道理了。到后來,能買到的書也多了,上網也更方便更便宜了......

  為了讓VCKBASE上的朋友,不再經歷我曾經的痛苦、不再重蹈我“無頭蒼蠅”般探索的艱辛、為了VCKBASE的蓬勃發展、為了中國軟件事業的騰飛(糟糕,吹的太也高了)......我打算節約一些在 BBS 上賺分的時間,寫個系列論文,就叫“COM組件設計與應用”吧。今天是第一部分——起源。

二、文件的存儲

  傳說350年前,牛頓被蘋果砸到了頭,于是發現了萬有引力。但到了二十一世紀的現在,任何一個技術的發明和發展,已經不再依靠圣人靈光的一閃。技術的進步轉而是被社會的需求、商業的利益、競爭的壓力、行業的滲透等推動的。微軟在Windows平臺上的組件技術也不例外,它的發明,有其必然因素。什么是這個因素那?答案是——文件的存儲。
  打開記事本程序,輸入了一篇文章后,保存。——這樣的文件叫“非結構化文件”;
  打開電子表格程序,輸入一個班的學生姓名和考試成績,保存?!@樣的文件叫“標準結構化文件”;
  在我們寫的程序中,需要把特定的數據按照一定的結構和順序寫到文件中保存。——這樣的文件叫“自定義結構化文件”;(比如 *.bmp 文件)
  以上三種類型的文件,大家都見的多了。那么文件存儲就依靠上述的方式能滿足所有的應用需求嗎?恩~~~,至少從計算機發明后的50多年來,一直是夠用的了。嘿嘿,下面看看商業利益的推動作用,對文件 的存儲形式產生了什么變化吧。30歲以上的朋友,我估計以前都使用過以下幾個聞名的軟件:WordStar(獨霸DOS下的英文編輯軟件),wps(裘伯君寫的中文編輯軟件,據說當年的市場占有率高達90%,各種計算機培訓班的必修課程),LOTUS-123(蓮花公司出品的電子表格軟件)......
微軟在成功地推出 Windows 3.1 后,開始垂涎桌面辦公自動化軟件領域。微軟的 Office 開發部門,各小組分別獨立地開發了 WORD 和 Excel 等軟件,并采用“自定義結構”方式,對文件進行存儲。在激烈的市場競爭下,為了打敗競爭對手,微軟自然地產生了一個念頭------假如我能在 WORD 程序中嵌入 EXCEL,那么用戶在購買了我 WORD 軟件的情況下,不就沒有必要再買 LOTUS-123 了嗎?!“惡毒”(中國微軟的同志們看到了這個詞,不要激動,我是加了引號的呀)的計劃產生后,他們開始了實施工作,這就是 COM 的前身 OLE 的起源(注3)。但馬上就碰到了一個嚴重的技術問題:需要把 WORD 產生的 DOC 文件和 EXCEL 產生的 XLS 文件保存在一起。
方案 優點 缺點 建立一個子目錄,把 DOC、XLS 存儲在這同一個子目錄中。 數據隔離性好,WORD 不用了解 EXCEL 的存儲結構;輕易擴展。 結構太松散,輕易造成數據的損壞或丟失。
不易攜帶。 修改文件存儲結構,在DOC結構基礎上擴展出包容 XLS 的結構。 結構緊密,輕易攜帶和統一治理。 WORD 的開發人員需要通曉 EXCEL 的存儲格式;缺少擴展性,總不能新加一個類型就擴展一下結構吧?!
以上兩個方案,都有嚴重的缺陷,怎么解決那?假如能有一個新方案,能夠合并前兩個方案的優點,消滅缺點,該多好呀......微軟是作磁盤操作系統起家的,于是很自然地他們提出了一個非常完美的設計方案,那就是把磁盤文件的治理方式移植到文件中了------復合文件,俗稱“文件中的文件系統”。連微軟當年都沒有想到,就這么一個簡單的想法,居然最后就演變出了 COM 組件程序設計的方法。可以說,復合文件是 COM 的基石。下圖是磁盤文件組織方式與復合文件組織方式的類比圖:
COM 組件設計與應用(一)起源及復合文件(圖一)圖一、左側表示一個磁盤下的文件組織方式,右側表示一個復合文件內部的數據組織方式。

三、復合文件的特點
  1. 復合文件的內部是使用指針構造的一棵樹進行治理的。編寫程序的時候要注重,由于使用的是單向指針,因此當做定位操作的時候,向后定位比向前定位要快;
  2. 復合文件中的“流對象”,是真正保存數據的空間。它的存儲單位為512字節。也就是說,即使你在流中只保存了一個字節的數據,它也要占據512字節的文件空間。啊~~~,這也太浪費了呀?不浪費!因為文件保存在磁盤上,即使一個字節也還要占用一個“簇”的空間那;
  3. 不同的進程,或同一個進程的不同線程可以同時訪問一個復合文件的不同部分而互不干擾;
  4. 大家都有這樣的體會,當需要往一個文件中插入一個字節的話,需要對整個文件進行操作,非常煩瑣并且效率低下。而復合文件則提供了非常方便的“增量訪問”能力;
  5. 當頻繁地刪除文件,復制文件后,磁盤空間會變的很零碎,需要使用磁盤整理工具進行重新整合。和磁盤治理非常相似,復合文件也會產生這個問題,在適當的時候也需要整理,但比較簡單,只要調用一個函數就可以完成了。
四、瀏覽復合文件

   VC6.0 附帶了一個工具軟件“復合文件瀏覽器”,文件名是“vc目錄/Common/Tools/DFView.exe”。為了方便使用該程序,可以把它加到工具(tools)菜單中。方法是:Tools/Customize.../Tools卡片中增加新的項目。運行 DFView.exe,就可以打開一個復合文件進行觀察了(注4)。但希奇的是,在 Microsoft Visual Studio .NET 2003 中,我反而找不到這個工具程序了,汗!不過這恰好提供給大家一個練習的機會,在你閱讀完本篇文章并把握了編程方法后,自己寫一個“復合文件瀏覽編輯器”程序,又練手了,還有實用的價值。

、復合文件函數

   復合文件的函數和磁盤目錄文件的操作非常類似。所有這些函數,被分為3種類型:WIN API 全局函數,存儲 IStorage 接口函數,流 IStream 接口函數。什么是接口?什么是接口函數?以后的文章中再陸續介紹,這里大家只要把“接口”看成是完成一組相關操作功能的函數集合就可以了。
  WIN API 函數 功能說明 StgCreateDocfile() 建立一個復合文件,得到根存儲對象 StgOpenStorage() 打開一個復合文件,得到根存儲對象 StgIsStorageFile() 判定一個文件是否是復合文件   IStorage 函數 功能說明 CreateStorage() 在當前存儲中建立新存儲,得到子存儲對象 CreateStream() 在當前存儲中建立新流,得到流對象 OpenStorage() 打開子存儲,得到子存儲對象 OpenStream() 打開流,得到流對象 CopyTo() 復制存儲下的所有對象到目標存儲中,該函數可以實現“整理文件,釋放碎片空間”的功能 MoveElementTo() 移動對象到目標存儲中 DestoryElement() 刪除對象 RenameElement() 重命名對象 EnumElements() 枚舉當前存儲中所有的對象 SetElementTimes() 修改對象的時間 SetClass() 在當前存儲中建立一個非凡的流對象,用來保存CLSID(注5) Stat() 取得當前存儲中的系統信息 Release() 關閉存儲對象   IStream 函數 功能說明 Read() 從流中讀取數據 Write() 向流中寫入數據 Seek() 定位讀寫位置 SetSize() 設置流尺寸。假如預先知道大小,那么先調用這個函數,可以提高性能 CopyTo() 復制流數據到另一個流對象中 Stat() 取得當前流中的系統信息 Clone() 克隆一個流對象,方便程序中的不同模塊操作同一個流對象 Release() 關閉流對象   WIN API 補充函數 功能說明 WriteClassStg() 寫CLSID到存儲中,同IStorage::SetClass() ReadClassStg() 讀出WriteClassStg()寫入的CLSID,相當于簡化調用IStorage::Stat() WriteClassStm() 寫CLSID到流的開始位置 ReadClassStm() 讀出WriteClassStm()寫入的CLSID WriteFmtUserTypeStg() 寫入用戶指定的剪貼板格式和名稱到存儲中 ReadFmtUserTypeStg() 讀出WriteFmtUserTypeStg()寫入的信息。方便應用程序快速判定是否是它需要的格式數據。 CreateStreamOnHGlobal() 內存句柄 HGLOBAL 轉換為流對象 GetHGlobalFromStream() 取得CreateStreamOnHGlobal()調用中使用的內存句柄
為了讓大家快速地瀏覽和把握基本方法,上面所列表的函數并不是全部,我省略了“事務”函數和未實現函數部分。更全面的介紹,請閱讀 MSDN。
下面程序片段,演示了一些基本函數功能和調用方法。
示例一:建立一個復合文件,并在其下建立一個子存儲,在該子存儲中再建立一個流,寫入數據。 void SampleCreateDoc() { ::CoInitialize(NULL); // COM 初始化 // 假如是MFC程序,可以使用AfxOleInit()替代 HRESULT hr; // 函數執行返回值 IStorage *pStg = NULL; // 根存儲接口指針 IStorage *pSub = NULL; // 子存儲接口指針 IStream *pStm = NULL; // 流接口指針 hr = ::StgCreateDocfile( // 建立復合文件 L"c://a.stg", // 文件名稱 STGM_CREATE STGM_WRITE STGM_SHARE_EXCLUSIVE, // 打開方式 0, // 保留參數 &pStg); // 取得根存儲接口指針 ASSERT( SUCCEEDED(hr) ); // 為了突出重點,簡化程序結構,所以使用了斷言。 // 在實際的程序中則要使用條件判定和異常處理 hr = pStg->CreateStorage( // 建立子存儲 L"SubStg", // 子存儲名稱 STGM_CREATE STGM_WRITE STGM_SHARE_EXCLUSIVE, 0,0, &pSub); // 取得子存儲接口指針 ASSERT( SUCCEEDED(hr) ); hr = pSub->CreateStream( // 建立流 L"Stm", // 流名稱 STGM_CREATE STGM_WRITE STGM_SHARE_EXCLUSIVE, 0,0, &pStm); // 取得流接口指針 ASSERT( SUCCEEDED(hr) ); hr = pStm->Write( // 向流中寫入數據 "Hello", // 數據地址 5, // 字節長度(注重,沒有寫入字符串結尾的/0) NULL); // 不需要得到實際寫入的字節長度 ASSERT( SUCCEEDED(hr) ); if( pStm ) pStm->Release();// 釋放流指針 if( pSub ) pSub->Release();// 釋放子存儲指針 if( pStg ) pStg->Release();// 釋放根存儲指針 ::CoUninitialize() // COM 釋放 // 假如使用 AfxOleInit(),則不調用該函數 } COM 組件設計與應用(一)起源及復合文件(圖二)
圖二、運行示例程序一后,使用 DFView.exe 打開觀察復合文件的效果圖

示例二:打開一個復合文件,枚舉其根存儲下的所有對象。 #include // ANSI、MBCS、UNICODE 轉換 void SampleEnum() { // 假設你已經做過 COM 初始化了 LPCTSTR lpFileName = _T( "c://a.stg" ); HRESULT hr; IStorage *pStg = NULL; USES_CONVERSION; // (注6) LPCOLESTR lpwFileName = T2COLE( lpFileName ); // 轉換T類型為寬字符 hr = ::StgIsStorageFile( lpwFileName ); // 是復合文件嗎? if( FAILED(hr) ) return; hr = ::StgOpenStorage( // 打開復合文件 lpwFileName, // 文件名稱 NULL, STGM_READ STGM_SHARE_DENY_WRITE, 0, 0, &pStg); // 得到根存儲接口指針 IEnumSTATSTG *pEnum=NULL; // 枚舉器 hr = pStg->EnumElements( 0, NULL, 0, &pEnum ); ASSERT( SUCCEEDED(hr) ); STATSTG statstg; while( NOERROR == pEnum->Next( 1, &statstg, NULL) ) { // statstg.type 保存著對象類型 STGTY_STREAM 或 STGTY_STORAGE // statstg.pwcsName 保存著對象名稱 // ...... 還有時間,長度等很多信息。請查看 MSDN ::CoTaskMemFree( statstg.pwcsName ); // 釋放名稱所使用的內存(注6) } if( pEnum ) pEnum->Release(); if( pStg ) pStg->Release(); }

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久成人国产精品| 久久99精品国产99久久6尤物| 欧美一区二区三区图| 亚洲mm色国产网站| 国产在线播放91| 91亚洲人电影| 亚洲精品国产免费| 91亚洲国产成人久久精品网站| 国产va免费精品高清在线| 久久天天躁狠狠躁老女人| 色av中文字幕一区| 国产精品都在这里| 欧美性少妇18aaaa视频| 2019日本中文字幕| 美日韩精品视频免费看| 亚洲成人av中文字幕| 亚洲视频视频在线| 91色琪琪电影亚洲精品久久| 91亚洲国产成人久久精品网站| 欧美激情a∨在线视频播放| 成人激情黄色网| 97久久精品视频| 俺去了亚洲欧美日韩| 国产精品对白刺激| …久久精品99久久香蕉国产| 亚洲电影第1页| 57pao成人国产永久免费| 欧美日韩精品二区| 欧美视频专区一二在线观看| 91精品国产乱码久久久久久蜜臀| 亚洲欧洲xxxx| 97在线视频免费| 日韩精品中文字幕有码专区| 深夜福利亚洲导航| 伊人成人开心激情综合网| 国产精品视频不卡| 岛国av在线不卡| 久久九九热免费视频| 日韩在线视频线视频免费网站| 国产视频精品一区二区三区| 欧美日韩国产成人高清视频| 综合国产在线观看| 欧洲亚洲妇女av| 日韩欧美精品中文字幕| 亚洲春色另类小说| 精品久久久久久久久久| 国产精品视频一区二区三区四| 亚洲精品一区二区网址| 欧美美女操人视频| 在线亚洲男人天堂| 色天天综合狠狠色| 国产91在线视频| 日韩欧美福利视频| 久久久久国产一区二区三区| 欧美国产日韩在线| 亚洲视频在线免费看| 大伊人狠狠躁夜夜躁av一区| 川上优av一区二区线观看| 91网站在线看| 亚洲一区二区三区四区视频| 久久精品国产成人精品| 欧美精品福利在线| 操人视频在线观看欧美| 欧美精品电影在线| 欧美精品生活片| 色偷偷91综合久久噜噜| 日韩av毛片网| 日韩欧美一区视频| 久久亚洲欧美日韩精品专区| 日韩最新中文字幕电影免费看| 亚洲国产欧美一区二区丝袜黑人| 伊人久久久久久久久久久久久| 懂色av影视一区二区三区| 日韩激情在线视频| 成人av色在线观看| 久久精品亚洲一区| 亚洲白拍色综合图区| 久久视频在线观看免费| 国产精品日韩欧美综合| 国产精品久久久久不卡| 日韩av在线影视| 最新国产成人av网站网址麻豆| 亚洲激情中文字幕| 欧美国产视频日韩| 欧美一级高清免费播放| 97在线视频精品| 在线视频欧美日韩精品| 欧美日韩中文在线观看| 国产一区视频在线播放| 最近2019好看的中文字幕免费| 国产成人a亚洲精品| 久久中文字幕视频| 国产精品视频地址| 亚洲精品一区中文| 亚洲女人被黑人巨大进入| 日韩精品免费在线视频观看| 中文字幕亚洲欧美日韩2019| 欧美性高潮床叫视频| 91国产精品电影| 日本韩国在线不卡| 亚洲性线免费观看视频成熟| 久久精品国产精品亚洲| 久久天堂av综合合色| 国产综合福利在线| 亚洲wwwav| 日韩欧美在线视频日韩欧美在线视频| 精品久久久久久久久久国产| 久久久久久久久久久亚洲| 国语自产精品视频在免费| 97国产一区二区精品久久呦| 久久久久久久色| 国产精品入口福利| 97超碰色婷婷| 一区二区三区视频观看| 欧美亚洲第一页| 国产日韩在线播放| 亚洲成人三级在线| 欧美日韩中文字幕日韩欧美| 美女撒尿一区二区三区| 亚洲日本中文字幕免费在线不卡| 成人午夜在线影院| 欧美精品videosex牲欧美| 欧美在线视频一区二区| 成人h视频在线观看播放| 国产剧情久久久久久| 一区二区欧美在线| 欧美高清视频一区二区| 国产在线拍偷自揄拍精品| 欧美日本国产在线| 欧美一级淫片aaaaaaa视频| 国产日产欧美a一级在线| 亚洲欧洲黄色网| 日韩中文字幕在线| 亚洲va久久久噜噜噜久久天堂| 亚洲三级 欧美三级| 国产一区二区美女视频| 欧美裸体xxxx极品少妇软件| 国产一区二区三区视频在线观看| 欧美在线性爱视频| 91精品国产91久久久久| 亚洲欧美激情四射在线日| 精品高清一区二区三区| 国产精品视频自在线| 国产精品亚洲综合天堂夜夜| 国产中文欧美精品| 在线日韩日本国产亚洲| 欧美日韩国产区| 亚洲综合在线小说| 91天堂在线视频| 亚洲免费视频一区二区| 日韩中文娱乐网| 精品国产1区2区| 亚洲已满18点击进入在线看片| 亚洲欧美一区二区三区四区| 成人免费视频网| 欧洲成人午夜免费大片| wwwwwwww亚洲| 秋霞成人午夜鲁丝一区二区三区| 中文字幕一区二区三区电影| 欧美在线视频导航| 欧美激情国产日韩精品一区18| 成人国产精品av| 欧美影院久久久| 亚洲自拍欧美色图|