使用MFC::CArchive

本文介绍如何使用MFC的CArchive进行对象的序列化和反序列化,通过实例演示了CPerson类对象的持久化过程及从文件中读取的过程。

转自:http://www.cnblogs.com/ylhome/archive/2010/03/15/1686558.html


使用MFC::CArchive

准确地说,CArchive也属于二进制变长数据,由于CArchive采用了CObject的RTTI特性,同时存储了对象类的信息,因此它对于CObject对象的持久化和反持久化来说,的确是"很好很强大"。

早在第3章我们结合CObject讨论过CArchive的相关知识,要想用上CArchive的强大功能,我们需要做的仅仅是将自己的类定义成支持自创建的CObject子类,并且覆盖默认的Serialize()函数。

现在动手

以下我们将准备一个示例,使用CArchive实现CPerson对象的持久化和反持久化。

选择【Win32】→【Win32项目】→【控制台程序】命令,创建ArchiveTest。

新建CPerson类,CPerson的定义如下:

 
  1. class CPerson : public CObject  
  2. {  
  3. DECLARE_SERIAL(CPerson)  
  4. private:  
  5.     CString _name;  
  6.     int _age;  
  7.     bool _gender;  
  8.     CString _words;  
  9. public:  
  10.     CPerson();  
  11.     CPerson(CString name, int age, bool gender = true);  
  12.     CString getName();  
  13.     CString getWords();  
  14.     void setWords(CString words);  
  15.     int getAge();  
  16.     bool isMale();  
  17.     void say();  
  18.     void say(CString msg);  
  19.     virtual void Serialize(CArchive& ar);  
  20. }; 

CPerson类的实现如下:

 
  1. #include "stdafx.h"  
  2. #include "mfc-person.h"  
  3.  
  4. IMPLEMENT_SERIAL(CPerson, CObject, 1)  
  5.  
  6. CPerson::CPerson()  
  7. {  
  8.     _name = _T("无名氏");  
  9.     _age = 0;  
  10.     _gender = true;  
  11. }  
  12.  
  13. CPerson::CPerson(CString name, int age, bool gender)  
  14. {  
  15.     _name = name;  
  16.     _age = age;  
  17.     _gender = gender;  
  18. }  
  19.  
  20. CString CPerson::getName()  
  21. {  
  22.     return _name;  
  23. }  
  24.  
  25. CString CPerson::getWords()  
  26. {  
  27.     return _words;  
  28. }  
  29.  
  30. void CPerson::setWords(CString words)  
  31. {  
  32.     _words = words;  
  33. }  
  34.  
  35. int CPerson::getAge()  
  36. {  
  37.     return _age;  
  38. }  
  39.  
  40. bool CPerson::isMale()  
  41. {  
  42.     return _gender;  
  43. }  
  44.  
  45. void CPerson::say()  
  46. {  
  47.     say(_words);  
  48. }  
  49.  
  50. void CPerson::say(CString msg)  
  51. {  
  52.     _tprintf(_T("%s: %s\r\n"), _name, msg);  
  53. }  
  54.  
  55. void CPerson::Serialize(CArchive& ar)  
  56. {  
  57.     if (ar.IsStoring())  
  58.     {  
  59.         ar << this->_name<<this->_age<<this->_gender << this->_words;  
  60.     }  
  61.     else 
  62.     {  
  63.         ar >> this->_name>>this->_age>>this->_gender >> this->_words;  
  64.     }  

修改主程序,在main()中创建两个CPerson对象,然后将其持久化到文件中,再将其从文件中反持久化出来,调用对象的方法,试试它们还是不是活的:

 
  1. #include "stdafx.h"  
  2. #include "mfc-person.h"  
  3. #include "ArchiveTest.h"  
  4.  
  5. int main()  
  6. {  
  7.     setlocale(LC_ALL, "chs");  
  8.  
  9.     //创建两个待写入的对象  
  10.     CPerson tong(_T("佟湘玉"), 28, false);  
  11.     tong.setWords(_T("额滴神啊..."));  
  12.  
  13.     CPerson bai(_T("白展堂"), 27, true);  
  14.     bai.setWords(_T("葵花点穴手!"));  
  15.  
  16.     //准备写入  
  17.     CFile oFile(_T("persons.archive"),CFile:: modeCreate|CFile::modeWrite);  
  18.     CArchive oar(&oFile, CArchive::store);  
  19.  
  20.     //序列化进去了  
  21.     oar << &tong << &bai;  
  22.     //oar.WriteObject(&tong);  
  23.     //oar.WriteObject(&bai);  
  24.  
  25.     oar.Close();  
  26.     oFile.Close();  
  27.  
  28.     //准备读取  
  29.     CFile iFile(_T("persons.archive"), CFile::modeRead);  
  30.     CArchive iar(&iFile, CArchive::load);  
  31.  
  32.     CPerson *p1, * p2;  
  33.  
  34.     //序列化出来了  
  35.     iar >> p1 >> p2;  
  36.     //p1 = iar.ReadObject(RUNTIME_CLASS(CPerson));  
  37.     //p2 = iar.ReadObject(RUNTIME_CLASS(CPerson));  
  38.  
  39.     //看看他们是不是活的*_*ii  
  40.     p1->say();  
  41.     p2->say();  
  42.  
  43.     delete p1;  
  44.     delete p2;  

运行结果如图6-19所示,可以看出,作为被序列化的CPerson对象,佟掌柜和老白完完全全活过来了。


 
图6-19  运行结果

比较好奇的读者可以查看一下其生成的二进制文件,这个persons.archive看起来比较乱,但根据运行结果中佟湘玉和白展堂说的两句话,我们不必担心该存储方式的准确性,CArchive生成的二进制内容如图6-20所示。


 
(点击查看大图)图6-20  CArchive生成的二进制内容

还有一个问题:为什么我们可以使用"oar << &tong << &bai"和"iar >> p1 >> p2"? 实际上CArchive包含了支持"CObject *"参数的流操作符:

 
  1. _AFX_INLINE CArchive& AFXAPI operator<<(CArchive& ar, const CObject* pOb)  
  2. {  
  3.     ar.WriteObject(pOb);  
  4.     return ar;  
  5. }  
  6.  
  7. _AFX_INLINE CArchive& AFXAPI operator>>(CArchive& ar, CObject*& pOb)  
  8. {  
  9.     pOb = ar.ReadObject(NULL);  
  10.     return ar;  

感兴趣的程序员可以究根求源,打开CArchive的源码,就会发现WriteObject()和ReadObject()会反过来调用CObject的Serialize()函数,调用流程如图6-21所示。

(点击查看大图)图6-21  CArchive的调用流程

这正是MFC的精妙之处,它让CPerson不知不觉地就具备了的持久化和反持久化的功能。

===========================================


CFile //创建/打开文件 CFile file; file.Open(_T("test.txt"),CFile::modeCreate|CFile::modeNoTruncate|CFile::modeReadWrite); 文件打开模式可组合使用,用“|”隔开,常用的有以下几种: CFile::modeCreate:以新建方式打开,如果文件不存在,新建;如果文件已存在,把该文件长度置零,即清除文件原有内容。 CFile::modeNoTruncate:以追加方式打开,如果文件存在,打开并且不将文件长度置零,如果文件不存在,会抛出异常。一般与CFile::modeCreate一起使用,则文件不存在时,新建一个文件;存在就进行追加操作。 CFile::modeReadWrite:以读写方式打开文件。 CFile::modeRead:只读。 CFile::modeWrite:只写。 //写入数据 CString strValue = "Hello World!"; file.Write(strValue,strValue.GetLength()); //追加数据 file.SeekToEnd(); //将指针移至文件末尾进行追加 file.Write(strValue,strValue.GetLength()); //关闭文件 file.Close(); CStdioFile CStdioFile是CFile的派生类,对文件进行流式操作,对于文本文件的读写很有用处,可按行读取写入。 //写入数据 CString strValue = "Hello World!"; file.WriteString(strValue); //读取数据 CString strRead; file.ReadString(strRead); 当文件存在多行数据需要逐行读取时,可用函数BOOL CStdioFile::ReadString(CString& rString),当遇到"\n "时读取截断,如果文件未读完,返回true,否则返回false。 //逐行读取文件内容,存入strRead while(file.ReadString(strRead)) { ...; } 各种关于文件的操作在程序设计中是十分常见,如果能对其各种操作都了如指掌,就可以根据实际情况找到最佳的解决方案,从而在较短的时间内编写出高效的代码,因而熟练的掌握文件操作是十分重要的。本文将对Visual C++中有关文件操作进行全面的介绍,并对在文件操作中经常遇到的一些疑难问题进行详细的分析。   1.文件的查找   当对一个文件操作时,如果不知道该文件是否存在,就要首先进行查找。MFC中有一个专门用来进行文件查找的类CFileFind,使用它可以方便快捷地进行文件的查找。下面这段代码演示了这个类的最基本使用方法。   CString strFileTitle;   CFileFind finder;   BOOL bWorking = finder.FindFile("C:\\windows\\sysbkup\\*.cab");   while(bWorking)   {   bWorking=finder.FindNextFile();   strFileTitle=finder.GetFileTitle();   }   2.文件的打开/保存对话框   让用户选择文件进行打开和存储操作时,就要用到文件打开/保存对话框。MFC的类CFileDialog用于实现这种功能。使用CFileDialog声明一个对象时,第一个BOOL型参数用于指定文件的打开或保存,当为TRUE时将构造一个文件打开对话框,为FALSE时构造一个文件保存对话框。   在构造CFileDialog对象时,如果在参数中指定了OFN_ALLOWMULTISELECT风格,则在此对话框中可以进行多选操作。此时要重点注意为此CFileDialog对象的m_ofn.lpstrFile分配一块内存,用于存储多选操作所返回的所有文件路径名,如果不进行分配或分配的内存过小就会导致操作失败。下面这段程序演示了文件打开对话框的使用方法。   CFileDialog mFileDlg(TRUE,NULL,NULL,   OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT|OFN_ALLOWMULTISELECT,   "All Files (*.*)|*.*||",AfxGetMainWnd());   CString str(" ",10000);   mFileDlg.m_ofn.lpstrFile=str.GetBuffer(10000);   str.ReleaseBuffer();   POSITION mPos=mFileDlg.GetStartPosition();   CString pathName(" ",128);   CFileStatus status;   while(mPos!=NULL)   {   pathName=mFileDlg.GetNextPathName(mPos);   CFile::GetStatus( pathName, status );   }   3.文件的读写   文件的读写非常重要,下面将重点进行介绍。文件读写的最普通的方法是直接使用CFile进行,如文件的读写可以使用下面的方法:   //对文件进行读操作   char sRead[2];   CFile mFile(_T("user.txt"),CFile::modeRead);   if(mFile.GetLength()<2)   return;   mFile.Read(sRead,2);   mFile.Close();   //对文件进行写操作   CFile mFile(_T("user.txt "), CFile::modeWrite|CFile::modeCreate);   mFile.Write(sRead,2);   mFile.Flush();   mFile.Close();   虽然这种方法最为基本,但是它的使用繁琐,而且功能非常简单。我向你推荐的是使用CArchive,它的使用方法简单且功能十分强大。首先还是用CFile声明一个对象,然后用这个对象的指针做参数声明一个CArchive对象,你就可以非常方便地存储各种复杂的数据类型了。它的使用方法见下例。   //对文件进行写操作   CString strTemp;   CFile mFile;   mFile.Open("d:\\dd\\try.TRY",CFile::modeCreate|CFile::modeNoTruncate|CFile::modeWrite);   CArchive ar(&mFile,CArchive::store);   ar<>strTemp;   ar.Close();   mFile.Close();   CArchive的 <> 操作符用于简单数据类型的读写,对于CObject派生类的对象的存取要使用ReadObject()和WriteObject()。使用CArchive的ReadClass()和WriteClass()还可以进行类的读写,如:   //存储CAboutDlg类   ar.WriteClass(RUNTIME_CLASS(CAboutDlg));   //读取CAboutDlg类   CRuntimeClass* mRunClass=ar.ReadClass();   //使用CAboutDlg类   CObject* pObject=mRunClass->CreateObject();   ((CDialog* )pObject)->DoModal();   虽然VC提供的文档/视结构中的文档也可进行这些操作,但是不容易理解、使用和管理,因此虽然很多VC入门的书上花费大量篇幅讲述文档/视结构,但我建议你最好不要使用它的文档。关于如何进行文档/视的分离有很多书介绍,包括非常著名的《Visual C++ 技术内幕》。   如果你要进行的文件操作只是简单的读写整行的字符串,我建议你使用CStdioFile,用它来进行此类操作非常方便,如下例。   CStdioFile mFile;   CFileException mExcept;   mFile.Open( "d:\\temp\\aa.bat", CFile::modeWrite, &mExcept);   CString string="I am a string.";   mFile.WriteString(string);   mFile.Close();  4.临时文件的使用   正规软件经常用到临时文件,你经常可以会看到C:\Windows\Temp目录下有大量的扩展名为tmp的文件,这些就是程序运行是建立的临时文件。临时文件的使用方法基本与常规文件一样,只是文件名应该调用函数GetTempFileName()获得。它的第一个参数是建立此临时文件的路径,第二个参数是建立临时文件名的前缀,第四个参数用于得到建立的临时文件名。得到此临时文件名以后,你就可以用它来建立并操作文件了,如:   char szTempPath[_MAX_PATH],szTempfile[_MAX_PATH];   GetTempPath(_MAX_PATH, szTempPath);   GetTempFileName(szTempPath,_T ("my_"),0,szTempfile);   CFile m_tempFile(szTempfile,CFile:: modeCreate|CFile:: modeWrite);   char m_char='a';   m_tempFile.Write(&m_char,2);   m_tempFile.Close();   5.文件的复制、删除等   MFC中没有提供直接进行这些操作的功能,因而要使用SDK。SDK中的文件相关函数常用的有CopyFile()、CreateDirectory()、DeleteFile()、MoveFile()。它们的用法很简单,可参考MSDN。 1,判断文件是否存在 access(filename,mode); 2,对于不同用途又不同的文件操作,其中API函数CreateFile()也是比较有用处理方式,对于巨型文件很合适的其他的楼上的大都说了,不重复了. [1]显示对话框,取得文件名 CString FilePathName; CFileDialog dlg(TRUE);///TRUE为OPEN对话框,FALSE为S***E AS对话框 if (dlg.DoModal() == IDOK) FilePathName=dlg.GetPathName(); 相关信息:CFileDialog 用于取文件名的几个成员函数: 假如选择的文件是C:\WINDOWS\TEST.EXE 则(1)GetPathName();取文件名全称,包括完整路径。取回C:\WINDOWS\TEST.EXE (2)GetFileTitle();取文件全名:TEST.EXE (3)GetFileName();取回TEST (4)GetFileExt();取扩展名EXE [2]打开文件 CFile file("C:\HELLO.TXT",CFile::modeRead);//只读方式打开 //CFile::modeRead可改为 CFile::modeWrite(只写), //CFile::modeReadWrite(读写),CFile::modeCreate(新建) 例子: { CFile file; file.Open("C:\HELLO.TXT",CFile::modeCreate|Cfile::modeWrite); . . . } [3]移动文件指针 file.Seek(100,CFile::begin);///从文件头开始往下移动100字节 file.Seek(-50,CFile::end);///从文件末尾往上移动50字节 file.Seek(-30,CFile::current);///从当前位置往上移动30字节 file.SeekToBegin();///移到文件头 file.SeekToEnd();///移到文件尾 [4]读写文件 读文件: char buffer[1000]; file.Read(buffer,1000); 写文件: CString string("自强不息"); file.Write(string,8); [5]关闭文件 file.Close();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值