Jsonlizer,一个把C++各类数据转成 Json 结构体的玩意儿
这段时间突发奇想,觉得可以弄一个Json和C++各种数据类型互转的工具,因为Json在进行数据储存的时候,有一些先天的优势,传统的C++的序列化方式是将数据序列化到流数据里面,而流数据是典型的串行结构(或则说是一维结构),因此,流数据对数据的位置特别的敏感,一旦序列化的元素有调整,就会导致原来存储的流数据完全失效。这一点在游戏开发时非常的麻烦,我们不得不将各个接口进行拆分,让接口对应其独自的流数据,做到局部的隔离。但是这个办法也不是万能,因为,一旦接口内某些元素顺序因为开始时过于简单而并没有做拆分时,隐藏的风险种子就种下了。随着时间推移,一旦需要更改这些元素时,就会发现被锁死了手脚!
Json具有结构化的特征,涵盖了原型,数组,对象等数据类型,用来做存储是非常理想的,如果项目中存储的数据用Json存储,然后再序列化到二进制流数据内,这样似乎就结合了两者的优点,结合了速度和可扩展性。而我自己写的CJson类本身就是可以序列化到流数据的。而且从流数据恢复的速度远比解析Json字符串的速度快几倍,那么现在欠缺的就是如何将C++数据转换成CJson类。
思考了一段时间,我意识到 Jsonlizer 和 Serializer 有本质的区别,一个是构建层次结构,一个是构建一维结构,我本来想尽可能的复用Serializer的代码,但我意识到,这是不可能的。我能利用的,可能仅仅是一部分最基本的,被称为 trait 模板代码。


而想复用Serialize函数的想法是不可能的。毕竟一个是结构化的数据,一个是一维数据。于是我改变了策略,重新构造了Jsonlizer。首先要做的是重写整个Jsonlizer的模板代码。这部分代码非常的关键,就是用于识别C++的各类数据。

这几个文件的代码如下:
#ifndef TAGPJSONLIZERRIMITIVE_H
#define TAGPJSONLIZERRIMITIVE_H
//Begin section for file tagJsonPrimitive.h
//TODO: Add definitions that you want preserved
//End section for file tagJsonPrimitive.hnamespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template <class A, class T, bool save>struct tagJsonPrimitive{//Begin section for si::tagJsonPrimitive//TODO: Add attributes that you want preserved//End section for si::tagJsonPrimitivepublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline static void Invoke(A & node, T & t){//TODO Auto-generated method stubnode.Bind( t );}}; //end struct tagJsonPrimitive} // end namespace atom#endif
原型非常简单,没什么好说的。
#ifndef TAGJSONLIZERINVALID_H
#define TAGJSONLIZERINVALID_H
//Begin section for file tagJsonInvalid.h
//TODO: Add definitions that you want preserved
//End section for file tagJsonInvalid.hnamespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template <class A, class T, bool save>struct tagJsonInvalid{//Begin section for si::tagJsonInvalid//TODO: Add attributes that you want preserved//End section for si::tagJsonInvalidpublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline static void Invoke(A & n, T & t){UNREFERENCED_PARAMETER(n);UNREFERENCED_PARAMETER(t);//TODO Auto-generated method stubprintf( "Jsonlization NOT support these keyword: mutable\n" );}}; //end struct tagJsonInvalid} // end namespace atom#endif
不支持的类型就是输出提示。
#ifndef TAGJSONLIZERARRAY_H
#define TAGJSONLIZERARRAY_H
//Begin section for file tagJsonArray.h
//TODO: Add definitions that you want preserved
//End section for file tagJsonArray.h
#include "../../../serialization/trait/array_trait.h"namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template <class A, class T, bool save>struct tagJsonArray{//Begin section for si::tagJsonArray//TODO: Add attributes that you want preserved//End section for si::tagJsonArraypublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline static void Invoke(A & jr, T & t){//TODO Auto-generated method stub// 读取或写入数组的长度;U32 bound = static_cast<U32>( array_trait<T>::bound );U32 limit = bound;CJson node = jr.GetCurrentNode();if( save == false ) {bound = static_cast<U32>( node.Length() );}// 确认数组的长度是否越界;bound = atom_min( bound, limit );// 读取或写入数组的元素;for( size_t i = 0; i < bound; ++ i ){CJson child( true );if( save == false ) {child = node[i];}jr.Push( child );jr.Bind( t[i] );jr.Pop();if( save ) {node.Push( child );}}}}; //end struct tagJsonArray} // end namespace atom#endif
数组开始有点东西了,其中能看到 Jsonlizer 有几个函数:Push,Pop,GetCurrentNode。这几个函数其实就是维护一个节点堆栈,堆栈内放入的是被操作的节点列表。顶部的节点就是最新需要被操作的节点。
#ifndef TAGJSONLIZERCLASS_H
#define TAGJSONLIZERCLASS_H
//Begin section for file tagJsonClass.h
//TODO: Add definitions that you want preserved
//End section for file tagJsonClass.h// 为序列化类而特别准备的缺省模板函数。
template<class A, class T>
inline void Jsonlize(A & jr, T & t, bool save)
{t.Jsonlize( jr, save );
}namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template <class A, class T, bool save>struct tagJsonClass{//Begin section for si::tagJsonClass//TODO: Add attributes that you want preserved//End section for si::tagJsonClasspublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline static void Invoke(A & jr, T & t){//TODO Auto-generated method stubJsonlize( jr, t, save );}}; //end struct tagJsonClass} // end namespace atom#endif
类的代码其实不复杂,因为类的代码需要用户自己完成。
#ifndef TAGJSONLIZER_H
#define TAGJSONLIZER_H
//Begin section for file tagJsonlizer.h
//TODO: Add definitions that you want preserved
//End section for file tagJsonlizer.h#include "../../../serialization/trait/is_primitive.h"
#include "../../../serialization/trait/is_array.h"
#include "../../../serialization/trait/is_class.h"
#include "../../../serialization/trait/is_pointer.h"
#include "../../../serialization/trait/if_else.h"
#include "../../../serialization/trait/degradation_trait.h"
#include "tagJsonPrimitive.h"
#include "tagJsonArray.h"
#include "tagJsonClass.h"
#include "tagJsonInvalid.h"namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template <class A, class T, bool B>struct tagJsonlizer{//Begin section for si::tagJsonlizer//TODO: Add attributes that you want preserved//End section for si::tagJsonlizerpublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline static void Jsonlize(A & json, T & data){//TODO Auto-generated method stubtypedef typename degradation_trait<T>::type C;typedef typenameif_else<is_array<C>::value,tagJsonArray<A, C, B>, typenameif_else<is_class<C>::value, tagJsonClass<A, C, B>, typenameif_else<is_primitive<C>::value,tagJsonPrimitive<A, C, B>,tagJsonInvalid <A, C, B>>::type >::type >::type Invoker;Invoker::Invoke( json, type_cast(data) );}}; //end struct tagJsonlizer} // end namespace atom#endif
这个就是Jsonlizer模板的入口了,接下来就是导入,导出类的代码。
#ifndef CJSONIMPORTER_H
#define CJSONIMPORTER_H
//Begin section for file CJsonImporter.h
//TODO: Add definitions that you want preserved
//End section for file CJsonImporter.h
#include "../../../Common.h"
#include "../../../interface/IEmbedInterface.h"
#include "../../../interface/IInterface.h"
#include "../../../interface/IJsonlizerRoot.h"
#include "../../../enumeration/INTERFACE_ID.h"
#include "../../../os/character/CCharset.h"
#include "../../stl/a_string.h"
#include "../../stl/a_wstring.h"
#include "../../hex/hex.h"
#include "../../tool/CInterface.h"
#include "tagJsonlizer.h"namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"class CJsonImporter : public IEmbedInterface{//Begin section for atom::CJsonImporter//TODO: Add attributes that you want preserved//End section for atom::CJsonImporterprivate://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"IInterface * nest;#ifdef _SHIPPING_//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"IReferencedInterface * cast;#endifpublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonImporter(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual ~CJsonImporter(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int IncRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int DecRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int GetRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual IInterface * QueryInterface(U32 iid); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual void SetNest(IInterface * nest); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline CJson GetCurrentNode(){// TODO Auto-generated method stubCJson result;CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {result = root -> GetCurrentNode();}return result;}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Push(CJson & node){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {root -> Push( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Pop(){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {root -> Pop();}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(bool & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<bool>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(char & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<U08>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I08 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<I08>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I16 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<I16>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I32 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<I32>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I64 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<I64>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U08 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<U08>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U16 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<U16>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U32 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<U32>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U64 & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<U64>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(float & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<float>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(double & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<double>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(a_string & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();value = static_cast<a_string>( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(a_wstring & value){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();a_string text = node;CCharset charset( text.c_str() );value = charset.FromUtf8.ToUnicode;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(void * buffer, U64 length){// TODO Auto-generated method stubif( buffer && length ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();a_string text = node;FromHex( text, buffer, length );}}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(T & value){//TODO Auto-generated method stubtagJsonlizer<CJsonImporter, T, false>::Jsonlize( *this, value );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(const char * name, T & value){//TODO Auto-generated method stubif( name ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();CJson child = node[name];root -> Push( child );Bind( value );root -> Pop();}}}}; //end class CJsonImporter} //end namespace atom#endif
#ifndef CJSONEXPORTER_H
#define CJSONEXPORTER_H
//Begin section for file CJsonExporter.h
//TODO: Add definitions that you want preserved
//End section for file CJsonExporter.h
#include "../../../Common.h"
#include "../../../interface/IEmbedInterface.h"
#include "../../../interface/IInterface.h"
#include "../../../interface/IJsonlizerRoot.h"
#include "../../../enumeration/INTERFACE_ID.h"
#include "../../../os/character/CCharset.h"
#include "../../stl/a_string.h"
#include "../../stl/a_wstring.h"
#include "../../hex/hex.h"
#include "../../tool/CInterface.h"
#include "tagJsonlizer.h"namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"class CJsonExporter : public IEmbedInterface{//Begin section for atom::CJsonExporter//TODO: Add attributes that you want preserved//End section for atom::CJsonExporterprivate://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"IInterface * nest;#ifdef _SHIPPING_//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"IReferencedInterface * cast;#endifpublic://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonExporter(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual ~CJsonExporter(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int IncRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int DecRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual int GetRef(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual IInterface * QueryInterface(U32 iid); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual void SetNest(IInterface * nest); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline CJson GetCurrentNode(){// TODO Auto-generated method stubCJson result;CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {result = root -> GetCurrentNode();}return result;}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Push(CJson & node){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {root -> Push( node );}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Pop(){// TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ) {root -> Pop();}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(bool & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(char & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I08 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I16 & value) {//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I32 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(I64 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U08 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U16 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U32 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(U64 & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(float & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(double & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(const char * value) {//TODO Auto-generated method stubif( value ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value;}}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(const wchar_t * value){//TODO Auto-generated method stubif( value ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();CCharset charset( value );a_string text = charset.ToUtf8;node = text.c_str();}}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(a_string & value) {//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();node = value.c_str();}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(a_wstring & value){//TODO Auto-generated method stubCInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();CCharset charset( value.c_str() );a_string text = charset.ToUtf8;node = text.c_str();}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Bind(void * buffer, U64 length){//TODO Auto-generated method stubif( buffer && length ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){a_string text = ToHex( buffer, length );CJson node = root -> GetCurrentNode();node = text.c_str();}}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(T & value){//TODO Auto-generated method stubtagJsonlizer<CJsonExporter, T, true>::Jsonlize( * this, value );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(const T & value){//TODO Auto-generated method stubBind( const_cast<T &>(value) );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(const char * name, T & value){//TODO Auto-generated method stubif( name ){CInterface<IJsonlizerRoot> root;if( root.Mount(this, IID_JSONLIZER_ROOT) ){CJson node = root -> GetCurrentNode();CJson child = node[name];root -> Push( child );Bind( value );root -> Pop();}}}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline void Bind(const char * name, const T & value){//TODO Auto-generated method stubBind( name, const_cast<T &>(value) );}}; //end class CJsonExporter} //end namespace atom#endif
导入导出的类有一个特殊的函数:
inline void Bind(const char * name, T & value)
这个函数带了一个name的参数,其实就是针对类的成员变量而设的。
#ifndef CJSONLIZER_H
#define CJSONLIZER_H
//Begin section for file CJsonlizer.h
//TODO: Add definitions that you want preserved
//End section for file CJsonlizer.h
#include "../../../Common.h"
#include "CJsonExporter.h"
#include "CJsonImporter.h"
#include "CJsonlizerRoot.h"
#include "../CJson.h"inline const char * last_dot(const char * value)
{const char * result(value);if( value ) {for( ;; ++ value ){if( *value == 0 ) break;if( *value == '.' ) {result = value + 1;}}} return result;
}#define JKV(V) make_pair(a_string(#V), V)
#define JBD(A,V) A.Bind(last_dot(#V), V)namespace atom
{//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"class CJsonlizer : public IInterface{//Begin section for atom::CJsonlizer//TODO: Add attributes that you want preserved//End section for atom::CJsonlizerprivate://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonExporter exporter;//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonImporter importer;//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonlizerRoot root;public://@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"CJsonlizer(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual ~CJsonlizer(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"virtual IInterface * QueryInterface(U32 iid); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"bool Assign(CJson & data); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"bool Obtain(CJson & data); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"void Clear(); //@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline CJson GetCurrentNode(){//TODO Auto-generated method stubroot.GetCurrentNode();}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Push(CJson & node){//TODO Auto-generated method stubroot.Push( node );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"inline void Pop(){//TODO Auto-generated method stubroot.Pop();}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline CJsonlizer & operator <<(const T & value){//TODO Auto-generated method stubexporter.Bind( value );return( * this );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline CJsonlizer & operator <<(const pair<a_string, T> & value){//TODO Auto-generated method stubexporter.Bind( value.first.c_str(), value.second );return( * this );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline CJsonlizer & operator >>(T & value){//TODO Auto-generated method stubimporter.Bind( value );return( * this );}//@generated "UML to C++ (com.ibm.xtools.transform.uml2.cpp.CPPTransformation)"template<class T>inline CJsonlizer & operator >>(pair<a_string, T> & value){//TODO Auto-generated method stubimporter.Bind( value.first.c_str(), value.second );return( * this );}}; //end class CJsonlizer} //end namespace atom#endif
最后就是最终的绑定接口了,接下来我展示一个简单的例子:
struct tagTest
{U32 index;a_string name;tagTest():index(0) {}~tagTest() {}
};struct tagTest2
{U32 index;tagTest value[4];tagTest2():index(0) {}~tagTest2() {}
};template<class A>
inline void Jsonlize(A & jr, tagTest & t, bool save)
{JBD( jr, t.index );JBD( jr, t.name );
}template<class A>
inline void Jsonlize(A & jr, tagTest2 & t, bool save)
{JBD( jr, t.index );JBD( jr, t.value );
}
准备了两个类,这两个类有聚合关系。然后我初始化这两个类,然后用Jsonizer将这个类导出成Json数据结构。
tagTest2 in, out;in.index = 1;in.value[0].index = 101;in.value[0].name = "101";in.value[1].index = 102;in.value[1].name = "102";in.value[2].index = 103;in.value[2].name = "103";in.value[3].index = 104;in.value[3].name = "104";CJsonlizer jr;jr << in;CJson json;jr.Obtain( json );a_string value = json.Stringify();printf( "%s\n", value.c_str() );
运行的结果如下:

其实并不困难。大家也可以试试,其中 trait 的那些代码可以去看看boost的序列化相关的代码。
相关文章:
Jsonlizer,一个把C++各类数据转成 Json 结构体的玩意儿
这段时间突发奇想,觉得可以弄一个Json和C各种数据类型互转的工具,因为Json在进行数据储存的时候,有一些先天的优势,传统的C的序列化方式是将数据序列化到流数据里面,而流数据是典型的串行结构(或则说是一维…...
Qt仿音乐播放器:设置窗口、部件属性
// 设置窗口标志 this->setWindowFlag(Qt::FramelessWindowHint); //此设置将窗口设置成无边框模式//设置窗口背景透明 this->setAttribute(Qt::WA_TranslucentBackground,true); attribute:属性 Translucent:半透明 Qt::WA_TranslucentBackgro…...
使用 .NET 6 或 .NET 8 上传大文件
如果您正在使用 .NET 6,并且它拒绝上传大文件,那么本文适合您。 我分享了一些处理大文件时需要牢记的建议,以及如何根据我们的需求配置我们的服务,并提供无限制的服务。 本文与 https://blog.csdn.net/hefeng_aspnet/arti…...
基于特征工程(pca分析)、小波去噪以及数据增强,同时采用基于注意力机制的BiLSTM、随机森林、ARIMA模型进行序列数据预测
本文采用特征工程(pca分析)、小波去噪以及数据增强,同时采用基于注意力机制的BiLSTM、随机森林、ARIMA模型进行序列数据预测 基于BILSTM(双向长短期记忆网络)、随机森林回归和ARIMA(自回归积分滑动平均&am…...
攻防世界 PHP2
开启场景 访问 /index.php,页面无变化 访问 /index.phps index.php 和 index.phps 文件之间的主要区别在于它们的文件扩展名。 index.php:这是一个标准的 PHP 文件,通常用于编写 PHP 代码。当用户访问 index.php 文件时,Web 服务器…...
主板idyy
import java.math.BigInteger; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; public class MachineCodeGenerator { // 获取主板ID(这需要根据操作系统具体实现) private static String getMotherboardID() {…...
轻松实现向量搜索:探索 Elastic-Embedding-Searcher 项目
随着人工智能和机器学习技术的飞速发展,向量搜索已成为数据检索的重要方式。尤其是在处理大规模文本数据时,传统的基于关键词的检索方式已经难以满足需求。为了优化检索性能并提升搜索精度,向量搜索成为了更加高效的解决方案。而在这一领域&a…...
flask后端开发(3):html模板渲染
目录 渲染模板html模板获取路由参数 gitcode地址: https://gitcode.com/qq_43920838/flask_project.git 渲染模板 这样就能够通过html文件来渲染前端,而不是通过return了 html模板获取路由参数...
逻辑控制语句
一、逻辑控制语句 条件判断 if循环 for、while 二、条件判断 if 1、语法 if 条件:条件为真的操作条件为真的操作 else:条件为假的操作条件为假的操作 data_01 int(input("数字: "))if data_01 > 10:print("ok!!!")print("正确!!!")prin…...
[OpenGL]使用 Compute Shader 实现矩阵点乘
一、简介 本文介绍了如何使用 OpenGL 中的 compute shader 进行矩阵相乘的并行运算。代码目标是,输入两个大小为 10*10 的矩阵 A 和 B,计算 A*B 的结果并存储到矩阵 C 中。 二、代码 0. 代码逻辑 1. 初始化 glfw, glad, 窗口 2. 初始化 compute shad…...
jangow-01-1.0.1靶机
靶机 ip:192.168.152.155 把靶机的网络模式调成和攻击机kali一样的网络模式,我的kali是NAT模式, 在系统启动时(长按shift键)直到显示以下界面 ,我们选第二个,按回车。 继续选择第二个,这次按 e 进入编辑页面 接下来,…...
MySQL 查询大偏移量(LIMIT)问题分析
大偏移量查询缓慢?LIMIT: 会进行两步操作 性能消耗在哪里了?OFFSET操作问题 2 LIMIT 操作 如何优化? 大偏移量查询缓慢? 示例:(假设age字段有索引) SELECT * FROM test WHERE age>18 LIMIT 10000000 ,10;分析MySQL的 LIMIT 10000000 , 10 LIMIT: 会进行两步操作 OFF…...
Docker、containerd、安全沙箱、社区Kata Containers运行对比
大家看了解决有意义、有帮助记得点赞加关注!!! containerd、安全沙箱和Docker三种运行对比。 本文通过对比三种运行时的实现和使用限制、部署结构,帮助您根据需求场景了解并选择合适的容器运行。 一、容器运行时实现和使用限制…...
使用npm包的工程如何引入mapboxgl-enhance/maplibre-gl-enhance扩展包
作者:刘大 前言 在使用iClient for MapboxGL/MapLibreGL项目开发中,往往会对接非EPSG:3857坐标系的地图,由于默认不支持,因此需引入mapboxgl-enhance/maplibre-gl-enhance扩展包。 在使用Vue等其他框架,通过npm包下载…...
【NIFI】实现ORACLE->ORACLE数据同步
【NIFI】实现ORACLE->ORACLE数据同步 需求 使用nifi实现 oracle->oracle 不同数据库之间的数据同步, 如果想实现 oracle->oracle技术有很多,例如使用oracle golden gate或者是kettle等,或者是使用oralce的dblink技术也能实现。当让…...
单例模式的写法
单例模式(Singleton Pattern)是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。常用于管理共享资源(如数据库连接、配置文件、线程池等)。在实际编码中,有多种实现单例模式的方法&…...
Selenium实践总结
1.使用显示等待而不是隐式等待 隐式等待可能会导致不可预测的测试行为,尤其是在动态 Web 应用程序中。显式等待,它允许您 等待特定条件发生后再继续测试,这种方法提供了更多的控制和可靠性。 WebDriverWait wait new WebDriverWait(drive…...
Python数据可视化小项目
英雄联盟S14世界赛选手数据可视化 由于本学期有一门数据可视化课程,课程结课作业要求完成一个数据可视化的小Demo,于是便有了这个小项目,课程老师要求比较简单,只要求熟练运用可视化工具展示数据,并不要求数据来源&am…...
Python毕业设计选题:基于python的白酒数据推荐系统_django+hive
开发语言:Python框架:djangoPython版本:python3.7.7数据库:mysql 5.7数据库工具:Navicat11开发软件:PyCharm 系统展示 管理员登录 管理员功能界面 用户管理 白酒管理 系统管理 看板展示 系统首页 白酒详情…...
SQL-leetcode-180. 连续出现的数字
180. 连续出现的数字 表:Logs -------------------- | Column Name | Type | -------------------- | id | int | | num | varchar | -------------------- 在 SQL 中,id 是该表的主键。 id 是一个自增列。 找出所有至少连续出现三次的数字。 返回的…...
SciencePlots——绘制论文中的图片
文章目录 安装一、风格二、1 资源 安装 # 安装最新版 pip install githttps://github.com/garrettj403/SciencePlots.git# 安装稳定版 pip install SciencePlots一、风格 简单好用的深度学习论文绘图专用工具包–Science Plot 二、 1 资源 论文绘图神器来了:一行…...
线程与协程
1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指:像函数调用/返回一样轻量地完成任务切换。 举例说明: 当你在程序中写一个函数调用: funcA() 然后 funcA 执行完后返回&…...
第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明
AI 领域的快速发展正在催生一个新时代,智能代理(agents)不再是孤立的个体,而是能够像一个数字团队一样协作。然而,当前 AI 生态系统的碎片化阻碍了这一愿景的实现,导致了“AI 巴别塔问题”——不同代理之间…...
SpringCloudGateway 自定义局部过滤器
场景: 将所有请求转化为同一路径请求(方便穿网配置)在请求头内标识原来路径,然后在将请求分发给不同服务 AllToOneGatewayFilterFactory import lombok.Getter; import lombok.Setter; import lombok.extern.slf4j.Slf4j; impor…...
Caliper 配置文件解析:config.yaml
Caliper 是一个区块链性能基准测试工具,用于评估不同区块链平台的性能。下面我将详细解释你提供的 fisco-bcos.json 文件结构,并说明它与 config.yaml 文件的关系。 fisco-bcos.json 文件解析 这个文件是针对 FISCO-BCOS 区块链网络的 Caliper 配置文件,主要包含以下几个部…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果。 方法一:使用 XML 的 <foreach> 标签ÿ…...
算法:模拟
1.替换所有的问号 1576. 替换所有的问号 - 力扣(LeetCode) 遍历字符串:通过外层循环逐一检查每个字符。遇到 ? 时处理: 内层循环遍历小写字母(a 到 z)。对每个字母检查是否满足: 与…...
计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...
