VC程序设计中可序列化类的实现(2)
}
else
{
ar>>m_VertexNum;
if(m_VertexNum!=0)
{
temp=m_TopVertex=new CMapVertex;
//链表中的元素需要在堆中进行分配内存空间,析构时会释放空间
m_TopVertex->Serialize(ar);
}
for(UINT i=1;i<m_VertexNum;i++)
{
temp->next=new CMapVertex;
temp=temp->next;//用缺省&&函数构造的节点的next为NULL
temp->Serialize(ar);//CMapVertex类已实现序列化
}
//其它数据读操作
}
CObject:: Serialize(ar);
}
实现细节:
本文的第二个例子模拟了MFC中链表集合类的序列化方法,在MFC的集合类中就是用类似的方法完成的。我们可以通过该例看出在对象的个数不固定的时候的方法,我们需要存放对象时先存放对象的数目,在读取的时候就可以按照存放的数目读取固定的对象。另外我们可以发现的一个十分有趣的事情时,我们上面所说的五项内容似乎与序列化的关系不大--只要获得文档类对象就可以完成基本的操作。为什么需要进行那么多项与主题"无关"的操作呢?其实如果我们如果仅需要完成简单的数据读写操作,我们的确没有必要做那些工作,但是通过那些工作我们可以较为简单的实现一些复杂的数据操作,比如在CObject类中已经实现了AssertValid,GetRuntimeClass,IsKindOf等类的判断操作,而且使用析取和插入运算符也必须完整地完成上述内容,更重要的是如果我们完成了那些操作就有可能在基于模板的集合类中运用自定义的类。
集合类是在MFC编程中实现较为复杂数据结构的方法之一,MFC提供了CMap、CArray、CList三种基于模板的集合类。集合类已经实现了数据的添加、删除、遍历以及序列化等操作,但是基于模板的集合类并不是可以引用所有的数据类型,如果我们想使用自定义的类作为集合类的节点类型的时候,我们就需要在完成自定类的时候完成规定的一系列操作,不过单单完成要求的5点内容并不能完成设计的要求,我们还需要完成对拷贝构造&&函数、operatror = &&函数的重载。在上面的类中我们就需要在定义中添加以下&&函数。
CMapVertex(CMapVertex& other);
CMapVertex& operator =(CMapVertex& other);
这样修改以后CMapVertex类就可以在模板类中使用(比如定义CList<CMapVertex, CMapVertex>),建立更加复杂的对象,而且可以使用模板类的序列化&&函数直接进行读写操作。
注:实践发现CMapVertex类中不可以使用CStirng类作为成员变量,如果使用在数据读写的时候会出现错误;而且在重载拷贝构造&&函数、operatror = &&函数时需要把关键成员全部包括,否则添加到链表或数组里的对象会有部分成员未被赋值。
使用方法:
经过序列化以后我们就可以在程序中建立该类的对象,在文档类的序列化&&函数可以获得CArchive对象实现自定义数据的读写操作。另外需要说明的是,CArchive类对析取和插入运算符的重载只支持下表中的数据 类型
WORD DWORD BYTE
int LONG Float
double CString CObject*
POINT and CPoint SIZE and CSize CTime and CtimeSpan
RECT and CRect COleCurrency ColeVariant
COleDateTime COleDateTimeSpan
如果我们需要对其他类型(如bool型)作就需要进行显式的强制类型转化其他类型(如BYTE)进行写操作并通过临时变量读操作。自定义类型既可以使用&&keyword=%D6%B8%D5%EB&Submit=+%CB%D1%CB%F7+">指针(作为参数使用析取和插入运算符操作)进行读写,也可以使用Serialize&&函数进行读写,二者如何区分使用呢?MSDN给我们的答案是,当操作对象使静态对象或则已经分配好内存&&keyword=%D6%B8%D5%EB&Submit=+%CB%D1%CB%F7+">指针的时候,使用Serialize&&函数;当操作对象是&&keyword=%D6%B8%D5%EB&Submit=+%CB%D1%CB%F7+">指针且没有动态分派内存时使用重载操作符。
注:Visual C++中可以使用bool和BOOL两种布尔变量但是它们的机制可能不完全相同,因为声明为BOOL型的时候可以用操作符进行读写操作,笔者认为BOOL与BYTE相同。