本节推荐一个解析json的库,之前推荐过解析xml,未来将会推荐更多,帮助大家解读常用语法。
json库,这里推荐github上开源的rapidjson
https://github.com/Tencent/rapidjson
最近在写某buss的时候发现需要json处理,在不同的上下文之间传递数据,而只允许key:value这种键值对传递,那当value为一个key:value结构或者多个的时候,怎么解决呢?
一种解决方案便是本文要说的json串设计及解析。
将key:value转为json所需要的string转义字符串,随后使用rapidjson在另一端解析即可,下面来阐述一下基本的使用操作。
只需要下载源码,随后将include文件夹引入项目中,头文件引入如下:
#include "rapidjson/document.h"
如果需要引入其他功能,就把doucument换成其他头文件即可,记得正文代码引入namespace。
https://github.com/Tencent/rapidjson
在json串中通常有一些类型,例如下面这个json串:
{
"a":1
}
a是string,1是int,怎么在rapidjson中得到呢?
在rapidjson中提供了GetType()方法,可以返回如下Type:
//! Type of JSON value
enum Type {
kNullType = 0, //!< null
kFalseType = 1, //!< false
kTrueType = 2, //!< true
kObjectType = 3, //!< object
kArrayType = 4, //!< array
kStringType = 5, //!< string
kNumberType = 6 //!< number
};
可以看到,基本上json所需要的类型都涉及了,简直碉堡了~
那么这对于解析字符串时判断类型至关重要~
下面将重点阐述解析操作:
使用Parse进行解析,方便快捷,Document支持操作符[],因此可以方便根据某个key拿到value。
const char* json = "{\"name\":\"小文\",\"like\":1,\"obj\":{\"school\":\"bd\"}}";
Document root;
root.Parse(json);
// 2. 利用 DOM 作出修改。
Value& s = root["like"];
解析所有孩子跟遍历很像,可以通过专门的迭代器进行遍历,每个结点就是个Value,不断遍历即可,下面将其封装成一个函数,可以拿到一个json串的所有结点。
int ParseObjAllChild(rapidjson::Value& root,map<string,string>& mp) {
string key_str, val_str;
for (rapidjson::Value::ConstMemberIterator itr=root.MemberBegin(); itr != root.MemberEnd(); itr++) {
rapidjson::Value key;
rapidjson::Value value;
rapidjson::Document::AllocatorType allocator;
key.CopyFrom(itr->name, allocator);
value.CopyFrom(itr->value, allocator);
if (!key.IsString() || !value.IsString()) return -1;
mp[key.GetString()] = value.GetString();
}
return 0;
}
rapidjson里面可以提供StringBuffer与Writer,我将实现封装到下面模板函数里面:
template<typename T>
string stringify(const T& o) {
StringBuffer sb;
Writer<StringBuffer> writer(sb);
o.Accept(writer);
return sb.GetString();
}