基于MFC實現(xiàn)類的序列化詳解
序列化是將程序中的對象以一種二進制格式存儲到存儲設備中(例如文本/數(shù)據(jù)庫等),以實現(xiàn)“永生”或隨意“流動”
首先定義一個要序列化的類people,其屬性有age、weight、height。
#pragma once
class people {
public:
int age;
int weight;
int height;
};
定義好后將該類添加到 項目名view.h 的類中,作為其屬性,并且是指針類型

將需要存取的類實現(xiàn)序列化
1.序列化的類必須從CObject派生,或是從CObject的派生類派生
#pragma once
//導入#include<afx.h>頭文件,使類能繼承CObject
#include<afx.h>
class people :public CObject {
public:
int age;
int weight;
int height;
};
2.為該類定義一個不帶參數(shù)的構造函數(shù)

3.在類聲明中使用DECLARE_SERIAL(類名)宏
#pragma once
#include<afx.h>
class people :public CObject {
public:
DECLARE_SERIAL(people);
people();
public:
int age;
int weight;
int height;
};
4.在類的實現(xiàn)中添加IMPLEMENT_SERIAL(類名,父類名,版本號)宏
#include "pch.h"
#include "people.h"
//第一個參數(shù)是類的名字,第二個參數(shù)是父類名字,第三個參數(shù)是版本號(一般版本號從1開始)
IMPLEMENT_SERIAL(people, CObject, 1);
people::people()
{
}
5.在類的實現(xiàn)中,通過類向導添加Serialize虛函數(shù)(也可也手寫)

MFC提供了CArchive類可以將對象數(shù)據(jù)保存到永久設備,比如磁盤文件。當應用程序重新啟動后,CArchive類可以幫助我們從磁盤文件讀取這些數(shù)據(jù),然后在內(nèi)存中重新構建對應的對象;這樣就使得我們的對象數(shù)據(jù)永久存在,該過程稱之為序列化(或者串行化)。
void people::Serialize(CArchive& ar)
{
if (ar.IsStoring())
{ // storing code
}
else
{ // loading code
}
}
CArchive類重載了輸入輸出運算符,在存取時就是利用運算符重載
void people::Serialize(CArchive& ar)
{
if (ar.IsStoring())
{ // storing code
//存數(shù)據(jù)
ar << age << weight << height;
}
else
{ // loading code
//讀數(shù)據(jù)
ar >> age >> weight >> height;
}
}
如果說類的屬性中有自定義類型的數(shù)組,比如下面這種,其中CPoint是MFC類型對象
#pragma once
#include<afx.h>
class people :public CObject {
public:
CArray<CPoint>m_point;
};
那么在 Serialize設置存取時可以安這樣
void people::Serialize(CArchive& ar)
{
if (ar.IsStoring())
{ // storing code
//存數(shù)據(jù)
ar << age << weight << height;
}
else
{ // loading code
//讀數(shù)據(jù)
ar >> age >> weight >> height;
}
/*直接使用數(shù)組調(diào)用該方法,因為數(shù)組已經(jīng)實現(xiàn)了這個方法,數(shù)組存放的元素是CPoint,CPoint
本省也是支持序列化*/
m_point.Serialize(ar);
}
在程序啟動時加載序列化
在Doc.cpp文件中找到序列化接口函數(shù)
void Cdraw3Doc::Serialize(CArchive& ar)
{
if (ar.IsStoring())
{
// TODO: 在此添加存儲代碼
}
else
{
// TODO: 在此添加加載代碼
}
}
從view類里拿到要序列化的數(shù)據(jù)people*
void Cdraw3Doc::Serialize(CArchive& ar)
{
//取出第一個view在鏈表中的位置
POSITION pos = GetFirstViewPosition();
//通過位置獲得該view
Cdraw3View* pView = (Cdraw3View* )GetNextView(pos);
if (ar.IsStoring())
{
//存數(shù)據(jù)
ar << pView->human;
}
else
{
//取數(shù)據(jù)
ar >> pView->human;
}
}
這樣我們每次保存時數(shù)據(jù)就可以以二進制序列化保存,每次打開時又可以打開上次保存的內(nèi)容
如果people*是一個數(shù)組的話,例如這樣

那么我們在用序列化存取時就要先存取數(shù)組長度,再循環(huán)存取數(shù)組的內(nèi)容
void Cdraw3Doc::Serialize(CArchive& ar)
{
//取出第一個view在鏈表中的位置
POSITION pos = GetFirstViewPosition();
//通過位置獲得該view
Cdraw3View* pView = (Cdraw3View* )GetNextView(pos);
//獲得數(shù)組長度
int cnt = pView->human.GetSize();
if (ar.IsStoring())
{
//存數(shù)據(jù)
ar << cnt;
for (int i = 0; i < cnt; i++) {
//循環(huán)存儲數(shù)組
ar << pView->human.GetAt(i);
}
}
else
{
//取長度
ar >> cnt;
for (int i = 0; i < cnt; ++i) {
//依次取數(shù)據(jù),加入到數(shù)組中
people* p;
ar >> p;
pView->human.Add(p);
}
}
}到此這篇關于基于MFC實現(xiàn)類的序列化詳解的文章就介紹到這了,更多相關MFC類的序列化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

