【MFC】串行化实现文档存储和读取功能

01、文章目录

02、什么叫串行化?

在面向对象的程序设计中,一般都是用二进制文件来保存文档资料。在VC++中控制和使用文件流的方法很多,MFC程序设计中常用的有两种方法:用CFile对象存储和读取文件;利用串行化存取文件。其中用CFile对象直接存储文档,存在着以下两个问题:一是过程繁琐,例如绘图系统中常存在大量的数据对象(直线对象、矩形对象等);另一个问题是功能受限,VC++为了程序集成的需要,设汁成了复合文档,可以把各种外来对象(如OLE对象的嵌入和链接)的内容存储到外部文件中并从外部文件中读取内容更新构造对象,而程序的设计者并不知道这些对象中需要存储数据的内容和格式,在这种情况下,直接利用CFile类就无能为力了。

既然已决定采用串行化的方法,就要明确串行化的概念:CArchive类用于以持久的二进制形式(通常是磁盘存储)来存储一个复杂的对象网络,用户可以把对象的内容存储到存储区中,也可以从存储区中读取内容重新构造对象。这个过程成为“串行化”,下面介绍串行化的几个基础知识点。

03、串行化基础接口

  1. CArchive

    构造函数:CArchive( CFile* pFile, UINT nMode, int nBufSize = 4096, void* lpBuf = NULL ),参数意义查阅MSDN。两个重要函数:IsStoring()和IsLoading(),以及>>和<<两个重载的操作符。

  2. 如何使自定义的类具有串行化功能呢?

    有以下五个步骤:

  • Deriving your class from CObject (or from some class derived from CObject).
  • Overriding the Serialize member function.
  • Using the DECLARE_SERIAL macro in the class declaration.
  • Defining a constructor that takes no arguments.
  • Using the IMPLEMENT_SERIAL macro in the implementation file for your class.
  1. CObArray::Serialize()
    CObArray类支持CObject类对象指针数组,它如同一个指针数组来管理直接或间接派生的CObject对象指针。当调用一个CObArray对象的Serialize()函数进行存储操作时,首先存储了数组的项数,然后循环对各个项目(各个对象)进行存储,进行了如下操作:

    1、将对象的信息写入文件。包括类的信息;版本信息;
    2、调用对象的Serialize成员函数,将对象的数据写入文件。

    而当进行读取操作时,首先读取了存储数组的项目数,然后将数组的项目初始化成需要的项目数,然后进行如下操作:

    1、从文件读取类的信息,动态生成相应类(CLine)的对象,存放对象的指针;
    2、调用对象的Serialize()成员函数,从文件读取对象数据初始化新生成的对象。

  2. Document/View结构

    1、在MFC中,文档类负责管理数据,提供保存和加载数据的功能。视类负责数据的显示,以及给用户提供对数据的编辑和修改功能。
    2、MFC给我们提供Document/View结构,将一个应用程序所需要的“数据处理与显示”的函数空壳都设计好了,这些函数都是虚函数我们可以在派生类中重写这些函数。有关文件读写的操作在CDocument的Serialize函数中进行,有关数据和图形显示的操作在CViewOnDraw函数中进行。我们在其派生类中,只需要去关注Serialize和OnDraw函数就可以了,其它的细节我们不需要去理会,程序就可以良好地运行。
    3、当我们按下“File/Open”,Application Framework会激活文件打开对话框,让你指定文件名,然后自动调CGraphicDoc::Serialize读取文件。Application Framework还会调用CGraphicView::OnDraw,传递一个显示DC,让你重新绘制窗口内容。MFC给我们提供Document/View结构,是希望我们将精力放在数据结构的设计和数据显示的操作上,而不要把时间和精力花费在对象与对象之间、模块与模块之间的通信上。
    4、一个文档对象可以和多个视类对象相关联,而一个视类对象只能和一个文档对象相关联。

  3. String Table中IDR_MAINFRAME字符串资源中各子串的含义

    1、CDocTemplate::windowTitle,主窗口标题栏上的字符串,MDI程序不需要指定,将以IDR_MAINFRAME字符串为默认值。
    2、CDocTemplate::docName,缺省文档的名称。如果没有指定,缺省文档的名称是无标题。
    3、CDocTemplate::fileNewName,文档类型的名称。如果应用程序支持多种类型的文档,此字符串将显示在"File/New"对话框中。如果没有指定,就不能够在"File/New"对话框处理这种文件。
    4、CDocTemplate::filterName,文档类型的描述和一个适用于此类型的通配符过滤器。这个字符串将出现在“File/Open”对话框中的文件类型列表框中。要和CDocTemplate::filterExt一起使用。
    5、CDocTemplate::filterExt,文档的扩展名。如果没有指定,就不能够在“File/Open”对话框中处理这种文档。要和CDocTemplate::filterName一起使用。
    6、CDocTemplate::regFileTypeId,如果你以::RegisterShellFileTypes向系统的注册表注册文件类型,此值会出现在HEY_CLASSES_ROOT之下成为其子项,并仅供Windows内部使用。如果没有指定,这种文件类型就无法注册。
    7、CDocTemplate::regFileTypeName,这也是存储在注册表中的文件类型名称。它会显示于程序中用以访问注册表的对话框内。

04、串行化实现文档存储与读写

  1. 实现一个可串行化的类(五个步骤)。特别注意的是改写Serialize()函数;
  2. 修改String Table中IDR_MAINFRAME字符串资源中子串,使其符合自己需求;
  3. 在CMyDoc类中增加一个CObArray的成员变量;修改CMyDoc::Serialize();如果需要保存CMyView类中的数据成员,可以利用一下代码获取CMyView的指针:POSITION pos=GetFirstViewPosition(); CGraphicView pView=(CGraphicView)GetNextView(pos);
  4. 新建和打开文档时,要注意销毁原来的数据。在DOC的DeleteContents虚函数中是好时机。代码如下:
int nCount;  
nCount=m_obArray.GetSize();  
for(int i=0;i<nCount;i++)  
{  
    delete m_obArray.GetAt(i);//释放指针指向的内存空间  
   //m_obArray.RemoveAt(i);//移除链表中的元素。嘿嘿,别搞错了。但在此处不能这样用,会导致非法操作。要用下面的方法  
  
}  
Array.RemoveAll();  

还有一种方法:

while(nCount--)  
{  
     delete m_obArray.GetAt(nCount);  
     m_obArray.RemoveAt(nCount);  
}

05、小结

关于MFC中的串行化知识,单靠这点是无法吃透的,还需自行在网上去了解,然后结合实践,做一下小例子,跑一跑,才能更好的明白什么叫串行化。

版权申明:转载请注明出处,谢谢!

©️2020 CSDN 皮肤主题: 护眼 设计师:闪电赇 返回首页