1 样本案例
1.1数据内容是一个数组
{
"success" : false,
"toReturn" : [
{
"createTime" : "20080806114526000+0800",
"createUser" : "张三"
}
],
"total" : 1
}
代码
#include <iostream>
#include <fstream>
#include "json/json.h"
using namesapce std;
void ParseJsonText()
{
string strJsonText ="{\"total\":1,\"toReturn\":[{\"createTime\":\"20080806114526000+0800\",\"createUser\":\"张三\"}],\"success\":false}";
Json::Reader reader;
Json::Value value;
if (NULL == reader.parse(strJsonText,value)) return;
std::string strTotalNumber =value["total"].asString();
Json::Value subValue =value["toReturn"];
if (subValue.isNull()) return;
size_t count = subValue.size();
for (size_t i = 0; i < count; i++)
{
std::string strCreateTime =subValue[i]["createTime"].asString();
std::string strCreateUser =subValue[i]["createUser"].asString();
}
}
void StoreJsonTextToFile()
{
string strJsonText ="{\"total\":1,\"toReturn\":[{\"createTime\":\"20080806114526000+0800\",\"createUser\":\"张三\"}],\"success\":false}";
Json::Reader reader;
Json::Value value;
if (NULL == reader.parse(strJsonText,value)) return;
ofstream ofs;
ofs.open("storefile.json");
ofs << value.toStyledString()<< endl;
ofs.close();
}
1.2数据内嵌json
"code" : "SheBeiLiXianGaoJingShangChuan",
"params" : {
"alarm_source" : "192.68.1.0",
"id" : "234",
"remark" : "fire alarm",
"time_alarm" : "2017-2-1 21:34:21"
}
代码
Json::Value jsonCode;
jsonCode["code"] ="SheBeiLiXianGaoJingShangChuan";
Json::Value jsonParams;
jsonParams["id"] ="234";
jsonParams["remark"] = "firealarm";
jsonParams["time_alarm"] ="2017-2-1 21:34:21";
jsonParams["alarm_source"] ="192.68.1.0";
jsonCode["params"] = jsonParams;
1.3 类型判断
jsoncpp判断某个字段是否存在的方法如下:
1)if(root["url"].type() != Json::nullValue)
2)if(value["sex"].isNull())
2注意事项
1)非法格式导致程序崩溃
jsoncpp是一个c++使用的json库,通过重载中括号[]来实现json的语法,但是由于c++
是一个强类型的语言,jsoncpp在实现过程中,使用了大量的断言,如果遇到类型不正确的
时候,就会强制断言,导致程序退出
例如,将如下的字符串传递给jsoncpp
std::string strContext ="success":false,"msg":"鎵句笉鍒版ā鏉縞ode:HuoQuSheBeiTongDaoShiJianChuo11"
Json::Reader read();
Json::Value root;
if (NULL == read.parse(strContext, root))return -1;
虽然字符串不是一个正确的json格式的字符串,但是一样也会解析出来,导致的结果就是不管使用什么样子的字段判断,都会导致程序崩溃退出,进入到断言中
解决的方案是指定严格的json报文解析规则
std::string strContext ="success":false,"msg":"鎵句笉鍒版ā鏉縞ode:HuoQuSheBeiTongDaoShiJianChuo11"
Json::Readerread(Json::Features::strictMode());
Json::Value root;
if (NULL == read.parse(strContext, root))return -1;
这样子就不会出现任何的异常
由于Jsoncpp解析非法json时,会自动容错成字符类型。对字符类型取下标时,会触发assert终止进程。
2)不完整的json格式检测机制
即使采用了Json::Features::strictMode(),指定严格的json报文解析规则,面对如下的返回值,如果使用json一样会解析,但是解析出来的结果不是期望值
{
"szstart_time": "2017-09-20 14:23:46"
"szend_time": "2017-09-30 11:16:22"
},
{
"szstart_time": "2017-09-20 14:23:46",
"szend_time": "2017-09-30 11:16:22"
}
上述的字符串,如果在首尾添加中括号[],就会变成合法的json字符串,但是jsoncpp并没有完全实现严格的报文校验机制,因此,还是解析出错误的结果。除了jsoncpp库之外,还有rapidjson,但是暂时不存在性能的问题,因此没有采用,并且由于事先知道可能是一个数组,因此预先添加中括号,变为合法的json数据
3)提示字符串中字符无效
jsoncpp 在处理 UTF-8 编码的字符串是没有问题,意思是可以正确的解析出键值对,在
VS调试情况下,返回的字符串如果包含中文,会提示:字符串中字符无效,这是因为在VS
调试过程中,只支持GBK编码的中文,因此如果需要进行字符串的查看,观察字符串的中文含义,可以先将字符串从UTF-8编码转换为GBK编码,然后就可以进行变量的监视
提供UTF-8转GBK
std::string UTF8ToGBK(const char* szUTF8)
{
intlen = MultiByteToWideChar(CP_UTF8, 0, szUTF8, -1, NULL, 0);
wchar_t*wstr = new wchar_t[len + 1];
memset(wstr,0, len + 1);
MultiByteToWideChar(CP_UTF8,0, szUTF8, -1, wstr, len);
len= WideCharToMultiByte(CP_ACP, 0, wstr, -1, NULL, 0, NULL, NULL);
char*str = new char[len + 1];
memset(str,0, len + 1);
WideCharToMultiByte(CP_ACP,0, wstr, -1, str, len, NULL, NULL);
if(wstr) delete[] wstr;
returnstr;
}