必须为希望串行化的每个类覆盖Serialize。被覆盖的Serialize首先必须调用基类的Serialize函数。
在类的声明中必须使用DECLARE_SERIAL宏,并且在类的执行过程中也必须使用IMPLEMENT_SERIAL宏。
使用CArchive::IsLoading或CArchive::IsStoring函数,用于决定是否装载或存储了档案文件。
步骤是
1.必须在创建一个CArchive对象之前,创建一个CFile对象。
2.创建CArchive对象,要把它附加给表示一个打开文件的类CFile(或派生类)的对象上。(注意CArchive没有基类)
方法是:直接调用其构造函数进行关联。CArchive(CFile*pFile,UINTnMode,int nBufSize=4096,void*lpBuf=NULL );
3.使用一个串行化类通常有一个Serialize成员函数。并且使用DECLARE_SERIAL和IMPLEMENT_SERIAL。
//在在头文件使用DECLARE_SERIA(类名)在源文件使用 IMPLEMENT_SERIAL(类名,基类,版本)
Serialize 是一个虚函数,virtual void Serialize(CArchive&ar); //自己重写该虚函数。
4.使用CArchive::IsLoading或CArchive::IsStoring函数,用于决定是否装载或存储了档案文件。
5.重载提取(>>)和插入(<<)是进行装载或存储操作。
archive 相当于文件,更近一步其实是文件之前的一个内存缓冲区。
DECLARE_SERIAL / IMPLEMENT_SERIAL 宏
DECLARE_SERIAL也包含了DECLARE_DYNCREATE,它用于创建对象.
在头文件使用DECLARE_SERIA(类名)
在源文件使用 IMPLEMENT_SERIAL( CMyClass, CObject, VERSIONABLE_SCHEMA | 2 )
//第一个参数是你的类名,第bai二个参数是基类名,第三个参数是个版本号,你可以随便写。
#define DECLARE_SERIAL(class_name) \
DECLARE_DYNCREATE(class_name) \
friend CArchive& AFXAPI operator>>(CArchive& ar, class_name* &pOb);
#define IMPLEMENT_SERIAL(class_name, base_class_name, wSchema) \
CObject* PASCAL class_name::CreateObject() \
{ return new class_name; } \
_IMPLEMENT_RUNTIMECLASS(class_name, base_class_name, wSchema, \
class_name::CreateObject) \
CArchive& AFXAPI operator>>(CArchive& ar, class_name* &pOb) \
{ pOb = (class_name*) ar.ReadObject(RUNTIME_CLASS(class_name)); \
return ar; } \
即DECLARE_SERIAL重载了>>操作符,见上面。所以相当于从cobject派生的类具备了ReadObject的功能。
另外DECLARE_SERIAL只重载了>>操作符,没有重载了 << 操作符.因为writeobject不需要cruntimeclass.但readobject需要,因为读完文件后要动态创建操作。
另外CArchive类也定义>> 和 <<
friend CArchive& AFXAPI operator<<(CArchive& ar, const CObject* pOb)
friend CArchive& AFXAPI operator>>(CArchive& ar, CObject*& pOb);
friend CArchive& AFXAPI operator>>(CArchive& ar, const CObject*& pOb);
这些重载运算符均定义在afx.inl文件中。
它大概有这么几个步骤:
1. 因为DECLARE_SERIAL重载了>>操作符,所以可以保证是调用CMessg类的>>函数.
2. >>函数实际上调用的是ar的ReadObject(CRuntimeClass*)函数
3. ReadObject首先从文件中读取类判断信息(可能是一个字符串,可能是一个类索引),得到Class对应的ClassName;
4. 程序的模块状态中有所有的RuntimeClass的列表,因此,查找对应的程序支持的RuntimeClass(对比ClassName),获得对应的RuntimeClass;
5. RuntimeClass中含有创建对象的方法CreateObject,调用它,创建对应的对象.这里,因为CreateObject实际就是 New 一个对象,类似 new CMessg; 所以,为了支持序列化,必须有没有参数的构造函数.
6. 创建对象之后,调用Seralize(ar),读入真正的对象的信息.
7. 将对象的指针返回.
8. pMessg就指向一个对应的对象了.