Welcome

首页 / 软件开发 / VC.NET / COM组件设计与应用(十七)——持续性

COM组件设计与应用(十七)——持续性2009-12-25 vckbase 杨老师一、前言

我们写程序,经常需要实现这样的需求:

例一、程序运行产生一个窗口,用户关闭的时候需要记录窗口的位置,以便下次运行时保持位置不变;

例二、由于程序运行时间很长,今天执行一部分,明天继续执行。那么在下次运行前要恢复前次的状态;

... ... ... ...

智慧的老师:以上这些需求,如何实现呢?

懵懂的学生:这个简单,只要在程序退出前提取必要的信息保存到文件中,下次运行时再从文件中读出来,设置一下就OK了。

智慧的老师:恩,不错,这位同学的思想值得表扬。

懵懂的学生:不好意思,这都要感谢老师的栽培,我对您的景仰如滔滔江水......

智慧的老师:别臭P了,我话还没有说完那......如果你需要提取和保存的信息很多,结构很复杂......怎么办?

懵懂的学生:也好办,我设计一个结构来记录这些信息。

智慧的老师:恩......不错。但如果这些信息提供方是别人写的模块,并且随着版本的不同还经常变化,你怎么办?

懵懂的学生:... ...

智慧的老师:解决这些问题的方法是---持续性。

二、原理

持续性,也叫永久性。组件方提供 IPersistXXX 接口,调用者(容器)提供存储介质,比如文件啦、内存啦、注册表啦、流啦、文本啦......啦啦拉。需要保存的时候,调用者通过 IPersistXXX::Save() 接口函数让组件去自己存储属性信息,而调用者根本不用关心存储格式和存储内容;需要还原状态的时候,调用者打开存储介质,然后同样调用 IPersistXXX::Load() 接口函数让组件自己去读取属性信息并完成初始化的设置。

目前,微软定义了如下各种类型的持续性接口,足够满足你的需求了。我们只要在自己写的组件中实现其中一个或几个持续性接口,那么调用者就可以按照统一的方式和我们的组件协商完成属性信息的保存和状态还原了。

持续性接口简要说明
IPersist所有持续性接口的根,下面的接口大多从它派生出来。这个接口很简单,只有一个函数 GetClassID()它返回组件的 CLSID 号,以便调用者能保存这个号为将来 CoCreateInstance() 启动组件用。实现这个函数也很简单,只要返回你组件中的 CLSID_XXX 即可,或者比较省事的方法是返回 GetObjectCLSID() 。

IPersistStream派生自 IPersist,并增加了4个函数,从流(IStream)中读写组件属性信息。

IsDirty()组件内部属性是否发生了变化。为调用者是否需要保存信息提供依据
Load()从 IStream 中读入信息,初始化组件属性
Save()把属性信息保存到 IStream 中
GetSizeMax()返回信息尺寸,以便调用者事先开辟空间
IPersistStreamInit派生自 IPersistStream,并再增加了一个函数 InitNew() 用来完成一个默认的组件属性初始化。这个持续性接口是最常用的,本文示例中就实现了该接口。

IPersistMemory和 IPersistStreamInit 类似,但使用的是内存块,而不是大小可变化的 IStream 流。
IPersistStorage和 IPersistStream 类似,但保存属性信息使用的是存储 IStorage,一个 IStorage 中可以有多个 IStream。
IPersistFile和 IPersistStream 类似,但存储介质为文件。
IPersistPropertyBag使用属性包(属性名、属性值)的文本方式保存信息。在 IE 浏览器中,HTML 嵌入 ActiveX 控件通常使用这个方法。在 HTML 中插入控件,<param name="属性名称" value="值"> 这样的形式你应该见过吧?!

在下一回的文章中,我们介绍这个接口。因为在 ActiveX 中,它太常用了。

IPersistPropertyBag2扩展了 IPersistPropertyBag 接口。提供了更丰富一些的属性管理用函数。
IPersistMoniker用于命名(moniker)存储和读取状态的持续性接口。
IPersistHistory运行于 IE 上,想在用户浏览 WEB 页面时存储和读取状态的持续性接口。