如何将STL结合到MFC (一)

原创 2002年02月16日 23:38:00

如何将STL结合到MFC ()<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

1.1 STL连续容器的序列化

 

              如何将STL方便的在MFC中利用起来, 的确是VC++程序员无法回避的问题, 我将结合自己实际工作中的一些做法和编程心得, 写出一个系列出来, 希望和大家好好交流和讨论.

              第一回我想探讨一下怎样将MFC的序列化功能利用到STL容器上, 也就是当STL容器作为类的成员如何将其序列化的问题.

1.1.1 MFC的做法

              我们知道MFC自己本身带有CARRAY, CLIST以及CMAP三个容器, 他们的父类都是COBJECT并且又都实现了void Serialize(CArchive & ar) 方法, 所以当这些容器作为类的成员变量时, 非常容易序列化例如当我们声明如下的成员时:

              CArray<int,int> m_aInt;

              CList<int, int> m_lInt;

              CMap<int, int, int, int> m_mInt;

序列化函数如下:

void CExampleDoc::Serialize(CArchive& ar)

{

              if (ar.IsStoring())

              {

              }

              else

              {

              }

             

              m_aInt.Serialize(ar);

              m_lInt.Serialize(ar);

              m_mInt.Serialize(ar);

 

}

可是当我们STL容器作为成员变量也需要被序列化时, 我们可以怎样做了?

1.1.2  序列

化一般的做法

我们知道STL容器分成连续的和非连续的两种. 非连续容器中插入对象的位置和对象的值相关, 所以应该单独来处理. 那么就先来谈谈连续容器的序列化.他们包括vector deque 以及list.

              其实序列化是一个相当简单的操作. 如果我们申明了一个成员如下:

              std::vector<int>     m_vInt;

              那么我们可以这样序列化它

void CExampleDoc::Serialize(CArchive& ar)

{

                            if (ar.IsStoring())

          {

                        ar << m_vInt.size();

                        for( vector<int>::iterator it = m_vInt.begin()

                      ; it != m_vInt.end()

                      ; ++it ) {

                                      ar << *it;

}

                            }

          else

                            {

                                          long nSize;

                                          ar >> nSize;

                                          m_vInt.resize(nSize);

                        for( vector<int>::iterator it = m_vInt.begin()

            ; it != m_vInt.end()

                      ; ++it ) {

                                      ar >> *it;

}

                            }

}

1.1.3  可以引入函数对象的做法

               可以看到这样针对每个容器对象中的每一个ITEM去做序列化的工作的确比较麻烦.

所以我们可以这样做首先定义一个函数对象:

 

template<class T> struct MySerialize

: public std::unary_function<T,void>

{

                            MySerialize(CArchive & ar) : m_ar(ar) { }

void operator() ( T & info)

{

        if (m_ar.IsStoring())

                                          {

                                                        m_ar << info;

                                          }

                                          else

                                          {

                                                        m_ar >> info;

                        }

             }

  CArchive & m_ar;

};

然后在序列化时就可以这样用了:

 

void CExampleDoc::Serialize(CArchive& ar)

{

                            if (ar.IsStoring())

          {

                        ar << m_vInt.size();

                            }

          else

                            {

                                          long nSize;

                                          ar >> nSize;

                                          m_vInt.resize(nSize);

                            }

                            for_each(m_vInt.begin(),m_vInt.end(), MySerialize< int>(ar));

}

 1.1.4   所有的工作一次完成的做法

但是, 这样还是存在着一部分游离的代码, 去处理容器大小的问题.

如何把那部分代码也结合进来呢? 其实, 可以再加入一个模版函数, 如下所示.

 

template<class TYPE, class CONTAINER_TYPE>

void ContainerSerialize(CArchive & ar, CONTAINER_TYPE  & con , TYPE  tmp)

{

 

              if (ar.IsStoring())

              {

                            ar << con.size();

              }

              else

              {

                            long nSize;

                            ar >> nSize;

                            con.resize(nSize);

              }

              for_each(con.begin(), con.end(), MySerialize<TYPE>(ar));

};

 

那么我们的序列化代码就可以这样写了:

 

void CExampleDoc::Serialize(CArchive& ar)

{

                            if (ar.IsStoring())

          {

          }

          else

                            {

                            }

                            ContainerSerialize(ar, m_vInt, int());

}

,在下一回里我将说明非连续容器的序列化方法.

 

如何将百度云推送结合到自己的项目(一)

第一讲,我将讲解使用百度云推送的前期准备工作,以及如何导入修改官方的PushDemo项目。 推送是什么?         你的手机安装了一个新闻客户端,(手机连接了网络)。每天早晨打开手机,就看见手机...
  • u010982006
  • u010982006
  • 2013年12月01日 16:04
  • 2884

让C++和Python使用同一个枚举

为了能快速修改程序的逻辑,我在C++ 程序中使用了 Python作为脚本。程序内部需要接收脚本传过来的一些不同类型的消息,在C++ 程序中使用了枚举来区分这些消息。于是编写脚本时想到:如何能在Pyth...
  • wsXiqiang
  • wsXiqiang
  • 2007年10月25日 19:53
  • 773

caffe框架在添加自己的MFC程序(上)

自从深度学习火起来以后,搞计算机视觉的不了解一点深度学习的知识都不还意思。以前的人脸识别还是LBP+svm的时代,突然出现了一个深度学习,大家都努力的奋斗。为了响应号召,我们利用cnn训练自己的网络然...
  • shakevincent
  • shakevincent
  • 2016年08月04日 09:01
  • 3373

C++中的STL和MFC

一、STL简介  STL(Standard Template Library,标准模板库)是惠普实验室开发的一系列软件的统称。它是由Alexander Stepanov、Meng Lee和Dav...
  • lansesl2008
  • lansesl2008
  • 2014年07月18日 18:51
  • 3376

C++中STL与MFC的关系

这两个解释通俗易懂的:   c++是一门编程语言,这门语言有它自己的标准和规范(比如有自己的语法)。  同样,针对C++这门语言,标准化组织又规定了相关的“程序库”,程序库中有各式各样的...
  • u010763324
  • u010763324
  • 2016年03月17日 15:08
  • 772

[请教] 如何将P6spy,sqlprofiler 结合到Spring+Hibernate中

applicationContext.xml                            spy.propertiesmodule.log=com.p6spy.engine.logging....
  • darkangel07
  • darkangel07
  • 2010年05月09日 01:06
  • 576

让你的工程支持MFC

要想使你的工程支持MFC,请按照以下步骤作:1:在你的stdafx.h的头文件中加入:#include 这是一个囊括了MFC的集合定义的头文件,有了它,你的工程就识别Cstring类了。2:在你的工程...
  • wei801004
  • wei801004
  • 2005年12月09日 16:08
  • 1170

STL字节对齐分配策略

今天无意中看了一下STL,发现STL中的空间配置器有两个等级
  • dai_jing
  • dai_jing
  • 2014年07月25日 14:26
  • 655

Opengl教程之读取stl文件并绘制在picturecontrol控件内

Opengl教程之读取stl并绘制在picturecontrol控件内 By Cracent 作为机械专业方向的人员,我们经常需要用SolidWorks构建模型。有些时候我们被要求将这些模型通过OPE...
  • Cracent
  • Cracent
  • 2016年04月03日 10:30
  • 6101

在C++中动态创建二维数组和三维数组

.h文件中定义成员函数: char **m_GCode = new char*[100]; char ***G = new char**[100]; .c文件中实现: G::G()//构造...
  • STL1634614466
  • STL1634614466
  • 2017年03月25日 08:41
  • 439
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:如何将STL结合到MFC (一)
举报原因:
原因补充:

(最多只允许输入30个字)