随笔-341  评论-2670  文章-0  trackbacks-0
    众所周知,boost里面有一个Serialization。不过我这个跟boost的有本质上的差别。boost的Serialization是通过重载操作符将一个对象与一个流进行读写,而我则通过记录类的成员变量来自动产生相应的读写动作

    不过我的Serialization库有两个缺点,就是效率不是特别高,因为我产生的是可以与XML进行互相转换的通用对象。虽然这并不是必需的,只是我的个人爱好罢了。第二个缺点是Serialization会产生侵入式的代码。使用方法如下:

    首先改造需要被Serialize的类,使之成为一个Serializable Object:
 1 class TestTree : public VL_SerializableObject
 2 {
 3 public:
 4     VInt                    IntegerMember;
 5     VUnicodeString            StringMember;
 6     VL_AutoPtr<TestTree>    LeftTree;
 7     VL_AutoPtr<TestTree>    RightTree;
 8 
 9     TestTree()
10     {
11         VL_REGISTER_SERIALIZABLE_FIELD(TestTree,IntegerMember);
12         VL_REGISTER_SERIALIZABLE_FIELD(TestTree,StringMember);
13         VL_REGISTER_SERIALIZABLE_FIELD(TestTree,LeftTree);
14         VL_REGISTER_SERIALIZABLE_FIELD(TestTree,RightTree);
15     }
16 };

    其次构造一个VL_Serializer并对这个类进行注册:
1 VL_Serializer Serializer;
2 VL_REGISTER_SERIALIZABLE_CLASS(&Serializer,TestTree);

    完成了!现在可以尝试将一个对象通过Serializer转换成XML:
 1 VL_AutoPtr<TestTree> Root=new TestTree;
 2 Root->IntegerMember=10;
 3 Root->StringMember=L"Root";
 4 
 5 VL_AutoPtr<TestTree> Left=new TestTree;
 6 Left->IntegerMember=20;
 7 Left->StringMember=L"Left";
 8 
 9 VL_AutoPtr<TestTree> Right=new TestTree;
10 Right->IntegerMember=30;
11 Right->StringMember=L"Right";
12 
13 Root->LeftTree=Left;
14 Root->RightTree=Right;
15 
16 VL_CommonObject SerializedObject=Serializer.Serialize(Root.Object());
17 VL_XMLDocument Document;
18 Document.GetRootElement()->SetName(L"SerializedObject");
19 SerializedObject.SaveToXML(Document.GetRootElement());
20 VL_FileStream Stream(FileName,VL_FileStream::vfomWrite);
21 Document.Save(&Stream,true);

    保存的那一段比较多,不过由于是死的,所以可以写一个函数来完成这些事情。让我们看看保存到文件里面的XML吧:
 1 <?xml version="1.0" encoding="gb2312" standalone="no" ?> 
 2 <SerializedObject>
 3     <vcoClass TypeName="TestTree">
 4         <vcoField FieldName="IntegerMember">
 5             <vcoInt Value="0A000000" /> 
 6         </vcoField>
 7         <vcoField FieldName="LeftTree">
 8             <vcoClass TypeName="TestTree">
 9                 <vcoField FieldName="IntegerMember">
10                     <vcoInt Value="14000000" /> 
11                 </vcoField>
12                 <vcoField FieldName="LeftTree">
13                     <vcoNull /> 
14                 </vcoField>
15                 <vcoField FieldName="RightTree">
16                     <vcoNull /> 
17                 </vcoField>
18                 <vcoField FieldName="StringMember">
19                     <vcoString Value="Left" /> 
20                 </vcoField>
21             </vcoClass>
22         </vcoField>
23         <vcoField FieldName="RightTree">
24             <vcoClass TypeName="TestTree">
25                 <vcoField FieldName="IntegerMember">
26                     <vcoInt Value="1E000000" /> 
27                 </vcoField>
28                 <vcoField FieldName="LeftTree">
29                     <vcoNull /> 
30                 </vcoField>
31                 <vcoField FieldName="RightTree">
32                     <vcoNull /> 
33                 </vcoField>
34                 <vcoField FieldName="StringMember">
35                     <vcoString Value="Right" /> 
36                 </vcoField>
37             </vcoClass>
38         </vcoField>
39         <vcoField FieldName="StringMember">
40             <vcoString Value="Root" /> 
41         </vcoField>
42     </vcoClass>
43 </SerializedObject>

    造这种东西出来是有特殊的目的的,并不打算将这个Serialization做成一个非常通用的Serialization。

    关键部分的代码如下:
    头文件:
  1 /*******************************************************************************
  2 Vczh Library++ 2.0
  3 数据结构::序列化
  4 开发者:陈梓瀚
  5 
  6 接口:
  7 类:
  8   VL_SerializableObject                        :可序列化对象基类
  9   VL_Serializer                                :序列化/反序列化处理器
 10 函数:
 11   VL_REGISTER_SERIALIZABLE_CLASS            :注册可序列化类型
 12   VL_REGISTER_SERIALIZABLE_FIELD            :注册可序列化成员变量
 13 *******************************************************************************/
 14 
 15 #ifndef VL_SERIALIZATION
 16 #define VL_SERIALIZATION
 17 
 18 #include "Data\VL_Data_List.h"
 19 #include "Data\VL_Data_Map.h"
 20 #include "VL_CommonData.h"
 21 
 22 namespace vl
 23 {
 24     namespace serialization
 25     {
 26         using namespace collection;
 27         using namespace commondata;
 28 
 29         class VL_Serializer;
 30 
 31         class VL_SerializableObject : public VL_Base
 32         {
 33         protected:
 34             class FieldSerializerBase : public VL_Base
 35             {
 36             public:
 37                 typedef VL_List<VL_AutoPtr<FieldSerializerBase> , false , FieldSerializerBase*>        List;
 38             protected:
 39                 VUnicodeString                    FFieldName;
 40 
 41             public:
 42                 FieldSerializerBase(VUnicodeString FieldName);
 43 
 44                 virtual void                    Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)=0;
 45             };
 46 
 47 #define DECLARE_PRIMITIVE_SERIALIZER(FIELD)                                                                                \
 48             template<typename _Type>                                                                                    \
 49             class FieldSerializer<_Type , FIELD> : public FieldSerializerBase                                            \
 50             {                                                                                                            \
 51             protected:                                                                                                    \
 52                 _Type*                            FObject;                                                                \
 53                 FIELD _Type::*                    FFieldPointer;                                                            \
 54             public:                                                                                                        \
 55                 FieldSerializer(_Type* Object , FIELD _Type::* FieldPointer , VUnicodeString FieldName)                    \
 56                     :FieldSerializerBase(FieldName)                                                                        \
 57                 {                                                                                                        \
 58                     FObject=Object;                                                                                        \
 59                     FFieldPointer=FieldPointer;                                                                            \
 60                 }                                                                                                        \
 61                                                                                                                         \
 62                 void Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)                                \
 63                 {                                                                                                        \
 64                     ClassObject.SetFieldValue(FFieldName,VL_CommonObject::CreatePrimitive(FObject->*FFieldPointer));    \
 65                 }                                                                                                        \
 66             }
 67 
 68             template<typename _Type , typename _Field>
 69             class FieldSerializer : public FieldSerializerBase
 70             {
 71             protected:
 72                 _Type*                            FObject;
 73                 _Field _Type::*                    FFieldPointer;
 74             public:
 75                 FieldSerializer(_Type* Object , _Field _Type::* FieldPointer , VUnicodeString FieldName)
 76                     :FieldSerializerBase(FieldName)
 77                 {
 78                     FObject=Object;
 79                     FFieldPointer=FieldPointer;
 80                 }
 81                 
 82                 void Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)
 83                 {
 84                     ClassObject.SetFieldValue(FFieldName,Serializer->Serialize(&(FObject->*FFieldPointer)));
 85                 }
 86             };
 87 
 88             DECLARE_PRIMITIVE_SERIALIZER(VBool);
 89             DECLARE_PRIMITIVE_SERIALIZER(VByte);
 90             DECLARE_PRIMITIVE_SERIALIZER(VInt);
 91             DECLARE_PRIMITIVE_SERIALIZER(VFloat);
 92             DECLARE_PRIMITIVE_SERIALIZER(VDouble);
 93             DECLARE_PRIMITIVE_SERIALIZER(VSize);
 94             DECLARE_PRIMITIVE_SERIALIZER(VChar);
 95             DECLARE_PRIMITIVE_SERIALIZER(VWChar);
 96             DECLARE_PRIMITIVE_SERIALIZER(VUnicodeString);
 97 
 98 #undef DECLARE_PRIMITIVE_SERIALIZER
 99 
100             template<typename _Type , typename _Field>
101             class FieldSerializer<_Type , _Field*> : public FieldSerializerBase
102             {
103             protected:
104                 _Type*                            FObject;
105                 _Field* _Type::*                FFieldPointer;
106             public:
107                 FieldSerializer(_Type* Object , _Field* _Type::* FieldPointer , VUnicodeString FieldName)
108                     :FieldSerializerBase(FieldName)
109                 {
110                     FObject=Object;
111                     FFieldPointer=FieldPointer;
112                 }
113                 
114                 void Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)
115                 {
116                     ClassObject.SetFieldValue(FFieldName,Serializer->Serialize(FObject->*FFieldPointer));
117                 }
118             };
119 
120             template<typename _Type , typename _Field>
121             class FieldSerializer<_Type , VL_AutoPtr<_Field>> : public FieldSerializerBase
122             {
123             protected:
124                 _Type*                            FObject;
125                 VL_AutoPtr<_Field> _Type::*        FFieldPointer;
126             public:
127                 FieldSerializer(_Type* Object , VL_AutoPtr<_Field> _Type::* FieldPointer , VUnicodeString FieldName)
128                     :FieldSerializerBase(FieldName)
129                 {
130                     FObject=Object;
131                     FFieldPointer=FieldPointer;
132                 }
133                 
134                 void Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)
135                 {
136                     ClassObject.SetFieldValue(FFieldName,Serializer->Serialize((FObject->*FFieldPointer).Object()));
137                 }
138             };
139 
140             template<typename _Type , typename _Field>
141             void AddFieldSerializer(_Type* Object , _Field _Type::* FieldPointer , VUnicodeString FieldName)
142             {
143                 FFieldSerializers.Add(new FieldSerializer<_Type,_Field>(Object,FieldPointer,FieldName));
144             }
145         private:
146             FieldSerializerBase::List            FFieldSerializers;
147         public:
148             VL_SerializableObject();
149             ~VL_SerializableObject();
150 
151             virtual void                        Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject);
152             virtual VBool                        Deserialize(VL_Serializer* Serializer , const VL_CommonObject& ClassObject);
153         };
154 
155         class VL_Serializer : public VL_Base
156         {
157         public:
158             class ClassSerializerBase : public VL_Base
159             {
160             public:
161                 typedef VL_List<VL_AutoPtr<ClassSerializerBase> , false , ClassSerializerBase*>        List;
162                 virtual VBool                    Accept(VL_SerializableObject* Object)=0;
163                 virtual VUnicodeString            GetClassTypeName()=0;
164             };
165 
166             template<typename _Type>
167             class ClassSerializer : public ClassSerializerBase
168             {
169             protected:
170                 VUnicodeString                    FClassTypeName;
171 
172             public:
173                 ClassSerializer(VUnicodeString ClassTypeName)
174                 {
175                     FClassTypeName=ClassTypeName;
176                 }
177 
178                 VBool Accept(VL_SerializableObject* Object)
179                 {
180                     return dynamic_cast<_Type*>(Object)!=0;
181                 }
182 
183                 VUnicodeString GetClassTypeName()
184                 {
185                     return FClassTypeName;
186                 }
187             };
188         protected:
189             ClassSerializerBase::List            FClassSerializers;
190         public:
191             VL_Serializer();
192             ~VL_Serializer();
193 
194             void                                RegisterClass(ClassSerializerBase* aClassSerializer);
195             ClassSerializerBase*                SelectClass(VL_SerializableObject* Object);
196             VL_CommonObject                        Serialize(VL_SerializableObject* Object);
197         };
198 
199         template<typename _Type>
200         void GenericDeserializerMethod(VL_Serializer* Serializer , const VL_CommonObject& Object , VL_SerializableObject*& Result)
201         {
202             Result=new _Type();
203             Result->Deserialize(Serializer,Object);
204         }
205 
206 #define VL_REGISTER_SERIALIZABLE_CLASS(SERIALIZER,CLASS)                                        \
207         do{                                                                                        \
208         (SERIALIZER)->RegisterClass(new VL_Serializer::ClassSerializer<CLASS>(L#CLASS));        \
209         }while(0)
210 
211 #define VL_REGISTER_SERIALIZABLE_FIELD(CLASS,FIELD)                                                \
212         do{                                                                                        \
213             AddFieldSerializer(this,&CLASS::FIELD,L#FIELD);                                        \
214         }while(0)
215     }
216 }
217 
218 #endif

    实现文件:
 1 #include "VL_Serialization.h"
 2 
 3 namespace vl
 4 {
 5     namespace serialization
 6     {
 7 
 8 /*********************************************************************************************************
 9 VL_SerializableObject
10 *********************************************************************************************************/
11 
12         VL_SerializableObject::FieldSerializerBase::FieldSerializerBase(VUnicodeString FieldName)
13         {
14             FFieldName=FieldName;
15         }
16 
17         VL_SerializableObject::VL_SerializableObject()
18         {
19         }
20 
21         VL_SerializableObject::~VL_SerializableObject()
22         {
23         }
24 
25         void VL_SerializableObject::Serialize(VL_Serializer* Serializer , VL_CommonObject& ClassObject)
26         {
27             for(VInt i=0;i<FFieldSerializers.GetCount();i++)
28             {
29                 FFieldSerializers[i]->Serialize(Serializer,ClassObject);
30             }
31         }
32 
33         VBool VL_SerializableObject::Deserialize(VL_Serializer* Serializer , const VL_CommonObject& ClassObject)
34         {
35             return false;
36         }
37 
38 /*********************************************************************************************************
39 VL_Serializer
40 *********************************************************************************************************/
41 
42         VL_Serializer::VL_Serializer()
43         {
44         }
45 
46         VL_Serializer::~VL_Serializer()
47         {
48         }
49 
50         void VL_Serializer::RegisterClass(ClassSerializerBase* aClassSerializer)
51         {
52             FClassSerializers.Add(aClassSerializer);
53         }
54 
55         VL_Serializer::ClassSerializerBase* VL_Serializer::SelectClass(VL_SerializableObject* Object)
56         {
57             for(VInt i=0;i<FClassSerializers.GetCount();i++)
58             {
59                 ClassSerializerBase* Class=FClassSerializers[i].Object();
60                 if(Class->Accept(Object))
61                 {
62                     return Class;
63                 }
64             }
65             return 0;
66         }
67 
68         VL_CommonObject VL_Serializer::Serialize(VL_SerializableObject* Object)
69         {
70             if(Object)
71             {
72                 ClassSerializerBase* Class=SelectClass(Object);
73                 if(Class)
74                 {
75                     VL_CommonObject ClassObject=VL_CommonObject::CreateClass(Class->GetClassTypeName());
76                     Object->Serialize(this,ClassObject);
77                     return ClassObject;
78                 }
79                 else
80                 {
81                     return VL_CommonObject::CreateError();
82                 }
83             }
84             else
85             {
86                 return VL_CommonObject::CreateNull();
87             }
88         }
89 
90     }
91 }
posted on 2008-11-21 09:09 陈梓瀚(vczh) 阅读(2822) 评论(9)  编辑 收藏 引用 所属分类: C++

评论:
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-21 09:14 | 陈梓瀚(vczh)
为什么只有一部分呢,因为还没有100%完成。先放上来Preview一下,接受批评……  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-21 17:50 | 空明流转的马甲
侵入式的解决方案没啥了,VC你还是写个C++的分析器,生成一个Serializer吧。嘎嘎。
在C++上Serializer程序员要做的事情实在是太多了,不像C#,依靠Reflection能搞掂无数的问题。。。还有就是Serializer还是能难处理Reference的情况,嘎嘎。  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-21 20:57 | 空明流转的马甲
ps,别用vco这样的名称修饰了,用namespace就是了。  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-21 22:55 | 陈梓瀚(vczh)
嘛,难道你不认为有了这样的一个框架之后,代码生成很容易写么,嘿嘿……  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-22 01:50 | 李侃
俺用模版库写了一个支持STL容器的Serializer ,内部可选择用zip压缩字节,以及加密解密字节,已经用了很久了,感觉很爽  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2008-11-22 03:59 | 肥仔
走上了一条阳春白雪的道路  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2009-01-10 20:31 | llxisdsh
序列化和反序列化的Buffer要支持attach才快速, 特别是在消息解析时
反序列化一定要能返回成功与否, 非常重要  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2009-01-10 21:02 | 陈梓瀚(vczh)
其实是返回了的。  回复  更多评论
  
# re: 我做的一个C++用的Serialization库(含部分源码) 2009-04-07 20:38 | rockeet
使用 febird.dataio,比 boost.serialization 快 50 倍
http://code.google.com/p/febird
http://blog.csdn.net/whinah   回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理