jsoncpp 用法
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\":\"張三
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;
}
jsoncpp 用法