序列化是指对象将其状态保存在一个字节流中,并用该字节流重建对象。实例化实现了文档数据的保存和装入的幕后工作,MFC通过序列化实现了应用程序的文档读写功能。
序列化的基本思想是:一个类应该能够对自己的成员变量的数据进行读写操作,对象可以通过读操作而重新创建。即对象可以将其当前状态(由其成员变量的值表示)写入永久性存储体(通常指磁盘)中,以后可以从永久性存储体中读取(载入)对象的状态,从而重建对象。类的对象自己应该具备将状态值写入磁盘或从磁盘中读出的方法(即成员函数),这种对象的保存和恢复过程称为序列化。
实现序列化的主要工作就是设计序列化的类。MFC应用程序实现类的序列化必须满足以下4个条件:(1)类必须直接或间接地从CObject类派生而来;(2)类必须定义一个不带参数的构造函数,当从磁盘文件载入文档时调用该构造函数来创建一个可序列化的对象,使用从文件中读出来的数据填充对象的成员变量;(3)在类的头文件中使用DECLARE_SERIAL宏,在类的实现文件中使用IMPLEMENT_SERIAL宏;(4)在自定义类中重载序列化成员函数Serialize。
下面是一个MFC 序列化应用例程。展现了如何将一组CBook类型的对象写入文件以及如何再从文件中读取,使“对象永续(object persistence)”。
//Book.h
#include <afxwin.h>
class CBook : public CObject
{
DECLARE_SERIAL(CBook) //写在类的声明部分,序列化支持宏
public:
CString m_strName;
CString m_strAuthor;
CString m_strPress;
public:
CBook()
{
// 要使CBook对象实现序列化,必须定义默认构造函数
}
CBook(CString strName, CString strAuthor, CString strPress)
{
m_strName = strName; // 书名
m_strAuthor = strAuthor; // 作者
m_strPress = strPress; // 出版社
}
void Serialize(CArchive &); // 重载基类的Serialize函数
};
//Book.cpp
#include "Book.h"
IMPLEMENT_SERIAL(CBook, CObject, 1)
void CBook::Serialize(CArchive &ar)
{
CObject::Serialize(ar);
if(ar.IsStoring())
{
ar << m_strName << m_strAuthor << m_strPress;
}
else
{
ar >> m_strName >> m_strAuthor >> m_strPress;
}
}
//主程序
CBook book1("Programming Windows", "Charles Petzold", "Microsoft Press");
CBook book2("Programming Windows with MFC", "Jeff Prosise", "Microsoft Press");
int nCount = 2;
// 序列化
CFile fileStore(_T("mybooks.dat"), CFile::modeWrite | CFile::modeCreate);
CArchive arStore(&fileStore, CArchive::store);
CBook* PBooksStore[2] = { &book1, &book2 };
arStore << nCount;
for (int i=0; i<nCount; ++i)
{
arStore << pBooksStore[i];
}
fileStore.Close();
// 反序列化
CFile fileLoad(_T("mybooks.dat"), CFile::modeRead);
CArchive arLoad(&fileLoad, CArchive::load);
arLoad >> nCount;
CBook* pBooksLoad = new CBook[nCount];
for (int i=0; i<nCount; ++i)
{
arLoad >> pBooksLoad[i];
}
fileLoad.Close();
MFC序列化注意事项:
·要想使某一类型能被序列化,在它类型声明文件(头文件,Book.h)和类型定义文件(Book.cpp)中要分别使用DECLARE_SERIAL和IMPLEMENT_SERIAL宏。
·能被序列化的只能指针型,不能是值型(pBooksStore数组中包含的就是两个CBook类型的指针)。