COM组件COMcomponent(COM组件)COM是微软公司为了计算机工业的软件生产更加符合人类的行为方式开发的一种新的软件开发技术。在COM构架下,人们可以开发出各种各样的功能专一的组件,然后将它们按照需要组合起来,构成复杂的应用系统。由此带来的好处是多方面的:可以将系统中的组件用新的替换掉,以便随时进行系统的升级和定制;可以在多个应用系统中重复利用同一个组件;可以方便的将应用系统扩展到网络环境下;COM与语言,平台无关的特性使所有的程序员均可充分发挥自己的才智与专长编写组件模块;等等。COM是开发软件组件的一种方法。组件实际上是一些小的二进制可执行程序,它们可以给应用程序,操作系统以及其他组件提供服务。开发自定义的COM组件就如同开发动态的,面向对象的API。多个COM对象可以连接起来形成应用程序或组件系统。并且组件可以在运行时刻,在不被重新链接或编译应用程序的情况下被卸下或替换掉。Microsoft的许多技术,如ActiveX,DirectX以及OLE等都是基于COM而建立起来的。并且Microsoft的开发人员也大量使用COM组件来定制他们的应用程序及操作系统。COM所含的概念并不止是在MicrosoftWindows操作系统下才有效。COM并不是一个大的API,它实际上象结构化编程及面向对象编程方法那样,也是一种编程方法。在任何一种操作系统中,开发人员均可以遵循“COM方法”。一个应用程序通常使由单个的二进制文件组成的。当编译器生成应用程序之后,在对下一个版本重新编译并发行新生成的版本之前,应用程序一般不会发生任何变化。操作系统,硬件及客户需求的改变都必须等到整个应用程序被重新生成。目前这种状况已经发生变化。开发人员开始将单个的应用程序分隔成单独多个独立的部分,也既组件。这种做法的好处是可以随着技术的不断发展而用新的组件取代以有的组件。此时的应用程序可以随新组件不断取代旧的组件而渐趋完善。而且利用已有的组件,用户还可以快速的建立全新的应用。传统的做法是将应用程序分割成文件,模块或类,然后将它们编译并链接成一个单模应用程序。它与组件建立应用程序的过程(称为组件构架)有很大的不同。一个组件同一个微型应用程序类似,即都是已经编译链接好并可以使用的二进制代码,应用程序就是由多个这样的组件打包而得到的。单模应用程序只有一个二进制代码模块。自定义组件可以在运行时刻同其他的组件连接起来以构成某个应用程序。在需要对应用程序进行修改或改进时,只需要将构成此应用程序的组件中的某个用新的版本替换掉即可。COM,即组件对象模型,是关于如何建立组件以及如何通过组件建立应用程序的一个规范,说明了如何可动态交替更新组件。使用组件的优点:组件架构的一个优点就是应用可以随时间的流逝而发展进化。除此之外,使用组件还有一些可以使对以有应用的升级更加方便和灵活的优点,如应用的定制,组件库以及分布式组件等。使用组件的种种优点直接来源于可以将它们动态的插入或卸出应用。为了实现这种功能,所有的组件必须满足两个条件:第一,组件必须动态链接;第二,它们必须隐藏(或封装)其内部实现细节。动态链接对于组件而言是一个至关重要的要求,而消息隐藏则是动态链接的一个必要条件。COM组件设计与应用(一)起源及复合文件作者:杨老师一、前言公元一九九五年某个夜黑风高的晚上,我的一位老师跟我说:“小杨呀,以后写程序就和搭积木一样啦。你赶快学习一些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(莲花公司出品的电子表格软件)......微软在成功地推出Windows3.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的基石。下图是磁盘文件组织方式与复合文件组织方式的类比图:图一、左侧表示一个磁盘下的文件组织方式,右侧表示一个复合文件内部的数据组织方式。三、复合文件的特点1.复合文件的内部是使用指针构造的一棵树进行管理的。编写程序的时候要注意,由于使用的是单向指针,因此当做定位操作的时候,向后定位比向前定位要快;2.复合文件中的“流对象”,是真正保存数据的空间。它的存储单位为512字节。也就是说,即使你在流中只保存了一个字节的数据,它也要占据512字节的文件空间。啊~~~,这也太浪费了呀?不浪费!因为文件保存在磁盘上,即使一个字节也还要占用一个“簇”的空间那;3.不同的进程,或同一个进程的不同线程可以同时访问一个复合文件的不同部分而互不干扰;4.大家都有这样的体会,当需要往一个文件中插入一个字节的话,需要对整个文件进行操作,非常烦琐并且效率低下。而复合文件则提供了非常方便的“增量访问”能力;5.当频繁地删除文件,复制文件后,磁盘空间会变的很零碎,需要使用磁盘整理工具进行重新整合。和磁盘管理非常相似,复合文件也会产生这个问题,在适当的时候也需要整理,但比较简单,只要调用一个函数就可以完成了。四、浏览复合文件VC6.0附带了一个工具软件“复合文件浏览器”,文件名是“vc目录\Common\Tools\DFView.exe”。为了方便使用该程序,可以把它加到工具(tools)菜单中。方法是:Tools\Customize...\Tools卡片中增加新的项目。运行DFView.exe,就可以打开一个复合文件进行观察了(注4)。但奇怪的是,在MicrosoftVisualStudio.NET2003中,我反而找不到这个工具程序了,汗!不过这恰好提供给大家一个练习的机会,在你阅读完本篇文章并掌握了编程方法后,自己写一个“复合文件浏览编辑器”程序,又练手了,还有实用的价值。五、复合文件函数复合文件的函数和磁盘目录文件的操作非常类似。所有这些函数,被分为3种类型:WINAPI全局函数,存储IStorage接口函数,流IStream接口函数。什么是接口?什么是接口函数?以后的文章中再陆续介绍,这里大家只要把“接口”看成是完成一组相关操作功能的函数集合就可以了。WINAPI函数功能说明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()关闭流对象WINAPI补充函数功能说明WriteClassStg()写CLSID到存储中,同IStorage::SetClass()ReadClassStg()读出WriteClassStg()写入的CLSID,相当于简化调用IStorage::Stat()WriteClassStm()写CLSID到流的开始位置ReadClassStm()读出WriteClassStm()写入的CLSIDWriteFmtUserTypeStg()写入用户指定的剪贴板格式和名称到存储中R