boost::serialization

简单的开始

Serialization的中文解释是“串行化” 、“序列化”或者“持久化” ,就是将内存中的对象保存到磁盘中,等到程序再次运行的时候再读取磁盘中的文件恢复原来的对象。下面来看一个简单的例子:

  1. #include <fstream>
  2. #include <iostream>
  3. #include <boost/archive/text_oarchive.hpp>
  4. #include <boost/archive/text_iarchive.hpp>
  5. class A
  6. {
  7. private:
  8.     //为了能让串行化类库能够访问私有成员,所以要声明一个友元类
  9.     friend class boost::serialization::access;
  10.     //对象的数据
  11.     int a;
  12.     double b;
  13.     //串行化的函数,这一个函数完成对象的保存与恢复
  14.     template<class Archive>
  15.     void serialize(Archive & ar, const unsigned int version)
  16.     {
  17.         ar & a;  //就是这么简单,也可以使用 ar<<a 这样的语法
  18.         ar & b;
  19.     }
  20. public:
  21.     A(int aa,double bb):a(aa),b(bb){}
  22.     A(){}
  23.     void print(){std::cout<<a<<' '<<b<<std::endl;}
  24. };  
  25. int main()
  26. {
  27.    std::ofstream fout("file.txt");//把对象写到file.txt文件中
  28.    boost::archive::text_oarchive oa(fout);//文本的输出归档类,使用一个ostream来构造
  29.    A obj(1,2.5);
  30.    oa<<obj;//保存obj对象
  31.    fout.close();//关闭文件
  32.  
  33.    std::ifstream fin("file.txt");
  34.    boost::archive::text_iarchive ia(fin);//文本的输入归档类
  35.    A newobj;
  36.    ia>>newobj;//恢复到newobj对象
  37.    newobj.print();
  38.    fin.close();
  39.    system("pause");
  40.    return 0;
  41. }

从上面可以看出,boost是使用text_oarchive和text_iarchive 类,来完成一个对象的序列化的。使用这两个类的步骤是:

  1. 在源程序中包含boost/archive/text_oarchive.hpp 和 boost/archive/text_iarchive.hpp 这两个文件。
  2. 为需要序列化的类添加一个template<class Archive> void serialize(Archive & ar, const unsigned int version)的模版成员函数。
  3. 如果需要对象中包含私有成员的话,需要把boost::serialization::access类声明为友元。
  4. 在主函数中,创建一个输出文件流对象, 使用这个对象构造一个text_oarchive对象,然后就可以使用<<操作符来输出对象了。
  5. 最后,同样的,使用text_iarchive来恢复对象。

继承

如果要序列化一个子类的话,方法是不同的。例:

  1. #include <boost/serialization/base_object.hpp> //一定要包含此头文件
  2. class B:A
  3. {
  4.     friend class boost::serialization::access;
  5.     char c;
  6.     template<class Archive>
  7.     void serialize(Archive & ar, const unsigned int version)
  8.     {
  9.         ar & boost::serialization::base_object<A>(*this);//注意这里
  10.         ar & c;
  11.     }
  12. public:
  13.     ...
  14. };   

对子类进行序列化的步骤是:

  1. 包含boost/serialization/base_object.hpp头文件
  2. 在serialize模版函数中,使用ar & boost::serialization::base_object<父类>(*this)这样的语法来保存父类的数据,不能直接调用父类的serialize函数

STL容器

如果要序列化一个STL容器,要使用boost自带的头文件,不能直接#include<vector>

例如:

  1. // Serialization中特定的头文件,在list.hpp中已经包含了stl的list头文件
  2. #include <boost/serialization/list.hpp>
  3. Class A
  4. {
  5.  ...
  6.  list<int> list;
  7. template<class Archive>
  8. void serialize(Archive & ar, const unsigned int version)
  9.  {
  10.       ar & list;
  11.  }
  12. ...
  13. }

在Serialization中,类似的头文件还有vector.hpp string.hpp set.hpp map.hpp slist.hpp等等。

数组和指针

对于数组和指针可以直接序列化,例:

  1. Class A
  2. {
  3.  ...
  4.  int a[10];
  5.  int *b
  6. template<class Archive>
  7. void serialize(Archive & ar, const unsigned int version)
  8.  {
  9.       ar & a;
  10.       ar & b;
  11.  }
  12. ...
  13. }

其他的archive类

除了text_iarchive和text_oarchive之外,还有其他的archive类,可以把对象保存成不同格式的文件。

  1. // a portable text archive
  2. boost::archive::text_oarchive(ostream &s) // saving
  3. boost::archive::text_iarchive(istream &s) // loading
  4.  
  5. // a portable text archive using a wide character stream
  6. boost::archive::text_woarchive(wostream &s) // saving
  7. boost::archive::text_wiarchive(wistream &s) // loading
  8.  
  9. // a non-portable native binary archive
  10. boost::archive::binary_oarchive(ostream &s) // saving
  11. boost::archive::binary_iarchive(istream &s) // loading
  12.  
  13. // a portable XML archive
  14. boost::archive::xml_oarchive(ostream &s) // saving
  15. boost::archive::xml_iarchive(istream &s) // loading
  16.  
  17. // a portable XML archive which uses wide characters - use for utf-8 output
  18. boost::archive::xml_woarchive(wostream &s) // saving
  19. boost::archive::xml_wiarchive(wistream &s) // loading

oost::serialization 基于 boost::archive 来完成任意复杂数据结构的序列化,boost::archive提供两个实现类来完成序列化、反序列化操



作:


boost::archive::text_oarchive     序列化数据,也称为:输出、保存(save)
boost::archive::text_iarchive      反序列化数据,也称为:输入、载入(load)


&操作符 


序列化操作使用 << 或者 & 操作符将数据存入text_oarchive中:


ar << data;
ar & data;


反序列化操作使用 >> 或者 & 操作符从text_iarchive中读取数据:


ar >> data;
ar & data;
为什么要引入&操作符?很简单,&操作符可以同时用于序列化和反序列化操作,这样就只需要提供一个 serialize 模板函数就可以同时用于两种


操作,具体执行哪种由ar的类型(模版参数类型)决定;如果ar是text_oarchive类型则是序列化,如果ar是text_iarchive类型则是反序列化。


serialize模版函数 


基础简单数据类型直接使用&操作符就行了,class类型数据需要自定义serialize模板函数来序列化对象,序列化时serialize函数会被自动调用





有两种方式来给class添加serialize接口,一种是成员函数方式(intrusive),另一种是非成员函数方式(non-intrusive)。这两种方式各有


利弊:non-intrusive方式虽然不用修改class的定义,但要求class数据成员的访问属性为public,不过可以通过声明函数为friend来解决这个


问题。


 成员函数方式(intrusive)

1 /**
 2 * GPS坐标点
 3 */
 4 class gps_position
 5 {
 6 private:
 7     friend class boost::serialization::access;
 8     
 9     //intrusive方式,该成员函数同时用于序列化和反序列化操作
10     template<class Archive>
11     void serialize(Archive & ar, const unsigned int version)
12     {
13         ar & degrees & minutes & seconds;
14     }
15 
16     int degrees;
17     int minutes;
18     float seconds;
19 
20 public:
21     gps_position(){};
22     gps_position(int d, int m, float s) :
23         degrees(d), minutes(m), seconds(s)
24     {}
25 };

非成员函数方式(non-intrusive) 

 1 /**
 2 * GPS坐标点
 3 */
 4 class gps_position
 5 {
 6     //non-intrusive方式,定义友元模版函数
 7     template<class Archive>
 8     friend void serialize(Archive& ar, gps_position& g, const unsigned int version)
 9     {
10         ar & g.degrees & g.minutes & g.seconds;
11     }
12 
13 
14 private:
15     int degrees;
16     int minutes;
17     float seconds;
18 
19 public:
20     gps_position(){};
21     gps_position(int d, int m, float s) :
22         degrees(d), minutes(m), seconds(s)
23     {}
24 };

类成员序列化 

对类进行序列化要求所涉及的数据成员必须是可序列化的,即为:简单数据类型(POD)、数组类型、STL容器、或者实现了serialize接口的用户

类。一个复合类类型实现了serialize接口后,就只需要将该类看做一个整体传给&操作符进行序列化,&操作符在序列化其数据成员时会自动调用

相应的serialize接口进行处理。

 1 /**
 2 * bus_stop类
 3 */
 4 class bus_stop
 5 {
 6     friend class boost::serialization::access;
 7 
 8     //只要数据成员类型是可序列化的就可以使用&操作符进行处理
 9     //可序列化的类型包括:基础的简单数据类型、数组类型、STL容器、或者实现了serialize接口的类
10     template<class Archive>
11     void serialize(Archive & ar, const unsigned int version)
12     {
13         ar & latitude & longitude;
14     }
15 
16     gps_position latitude;        //serializable members
17     gps_position longitude;
18 
19 protected:
20     bus_stop(const gps_position & lat_, const gps_position & long_) :
21     latitude(lat_), longitude(long_)
22     {}
23 
24 public:
25     bus_stop(){}
26     
27     virtual ~bus_stop(){}
28 };

编译程序

编译程序时需要带上库:-lboost_serialization
 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值