JSON(JavaScript Object Notation)跟xml一样也是一种数据交换格式,了解json请参考其官网http://json.org/,本文不再对json做介绍,将重点介绍c++的json解析库的使用方法。json官网上列出了各种语言对应的json解析库,作者仅介绍自己使用过的两种C++的json解析库:jsoncpp(v0.5.0)和Boost(v1.34.0)。

一. 使用jsoncpp解析json
Jsoncpp是个跨平台的开源库,首先从http://jsoncpp.sourceforge.net/上下载jsoncpp库源码,我下载的是v0.5.0,压缩包大约107K,解压,在jsoncpp-src-0.5.0/makefiles/vs71目录里找到jsoncpp.sln,用VS2003及以上版本编译,默认生成静态链接库。 在工程中引用,只需要include/json及.lib文件即可。

使用JsonCpp前先来熟悉几个主要的类:

Json::Value     可以表示里所有的类型,比如int,string,object,array等,具体应用将会在后边示例中介绍。

Json::Reader   将json文件流或字符串解析到Json::Value, 主要函数有Parse。

Json::Writer    与Json::Reader相反,将Json::Value转化成字符串流,注意它的两个子类:Json::FastWriter和Json::StyleWriter,分别输出不带格式的json和带格式的json。

1. 从字符串解析json

[cpp] view plaincopyprint?
int ParseJsonFromString()  
{  
  const char* str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}";  
  
  Json::Reader reader;  
  Json::Value root;  
  if (reader.parse(str, root))  // reader将Json字符串解析到root,root将包含Json里所有子元素   
  {  
    std::string upload_id = root["uploadid"].asString();  // 访问节点,upload_id = "UP000000"   
    int code = root["code"].asInt();    // 访问节点,code = 100   
  }  
  return 0;  
}  
int ParseJsonFromString()
{
  const char* str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}";

Json::Reader reader;
  Json::Value root;
  if (reader.parse(str, root))  // reader将Json字符串解析到root,root将包含Json里所有子元素
  {
    std::string upload_id = root["uploadid"].asString();  // 访问节点,upload_id = "UP000000"
    int code = root["code"].asInt();    // 访问节点,code = 100
  }
  return 0;
}
2. 从文件解析json

json文件内容:

[cpp] view plaincopyprint?
{  
    "uploadid": "UP000000",  
    "code": "0",  
    "msg": "",  
    "files":  
    [  
        {  
            "code": "0",  
            "msg": "",  
            "filename": "1D_16-35_1.jpg",  
            "filesize": "196690",  
            "width": "1024",  
            "height": "682",  
            "images":  
            [  
                {  
                    "url": "fmn061/20111118",  
                    "type": "large",  
                    "width": "720",  
                    "height": "479"  
                },  
                {  
                    "url": "fmn061/20111118",  
                    "type": "main",  
                    "width": "200",  
                    "height": "133"  
                }  
            ]  
        }  
    ]  
}  
{
    "uploadid": "UP000000",
    "code": "0",
    "msg": "",
    "files":
    [
        {
            "code": "0",
            "msg": "",
            "filename": "1D_16-35_1.jpg",
            "filesize": "196690",
            "width": "1024",
            "height": "682",
            "images":
            [
                {
                    "url": "fmn061/20111118",
                    "type": "large",
                    "width": "720",
                    "height": "479"
                },
                {
                    "url": "fmn061/20111118",
                    "type": "main",
                    "width": "200",
                    "height": "133"
                }
            ]
        }
    ]
}
 解析代码:

[cpp] view plaincopyprint?
int ParseJsonFromFile(const char* filename)  
{  
  // 解析json用Json::Reader   
  Json::Reader reader;  
  // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array...   
  Json::Value root;         
  
  std::ifstream is;  
  is.open (filename, std::ios::binary );    
  if (reader.parse(is, root))  
  {  
    std::string code;  
    if (!root["files"].isNull())  // 访问节点,Access an object value by name, create a null member if it does not exist.   
      code = root["uploadid"].asString();  
      
    // 访问节点,Return the member named key if it exist, defaultValue otherwise.   
    code = root.get("uploadid", "null").asString();  
  
    // 得到"files"的数组个数   
    int file_size = root["files"].size();  
  
    // 遍历数组   
    for(int i = 0; i < file_size; ++i)  
    {  
      Json::Value val_image = root["files"][i]["images"];  
      int image_size = val_image.size();  
      for(int j = 0; j < image_size; ++j)  
      {  
        std::string type = val_image[j]["type"].asString();  
        std::string url = val_image[j]["url"].asString();  
      }  
    }  
  }  
  is.close();  
  return 0;  
}  
int ParseJsonFromFile(const char* filename)
{
  // 解析json用Json::Reader
  Json::Reader reader;
  // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array...
  Json::Value root;

std::ifstream is;
  is.open (filename, std::ios::binary );  
  if (reader.parse(is, root))
  {
    std::string code;
    if (!root["files"].isNull())  // 访问节点,Access an object value by name, create a null member if it does not exist.
      code = root["uploadid"].asString();
    
    // 访问节点,Return the member named key if it exist, defaultValue otherwise.
    code = root.get("uploadid", "null").asString();

// 得到"files"的数组个数
    int file_size = root["files"].size();

// 遍历数组
    for(int i = 0; i < file_size; ++i)
    {
      Json::Value val_image = root["files"][i]["images"];
      int image_size = val_image.size();
      for(int j = 0; j < image_size; ++j)
      {
        std::string type = val_image[j]["type"].asString();
        std::string url = val_image[j]["url"].asString();
      }
    }
  }
  is.close();
  return 0;
}
 3. 在json结构中插入json

[cpp] view plaincopyprint?
Json::Value arrayObj;   // 构建对象   
Json::Value new_item, new_item1;  
new_item["date"] = "2011-12-28";  
new_item1["time"] = "22:30:36";  
arrayObj.append(new_item);  // 插入数组成员   
arrayObj.append(new_item1); // 插入数组成员   
int file_size = root["files"].size();  
for(int i = 0; i < file_size; ++i)  
  root["files"][i]["exifs"] = arrayObj;   // 插入原json中  
    Json::Value arrayObj;   // 构建对象
    Json::Value new_item, new_item1;
    new_item["date"] = "2011-12-28";
    new_item1["time"] = "22:30:36";
    arrayObj.append(new_item);  // 插入数组成员
    arrayObj.append(new_item1); // 插入数组成员
    int file_size = root["files"].size();
    for(int i = 0; i < file_size; ++i)
      root["files"][i]["exifs"] = arrayObj;   // 插入原json中
 4. 输出json

[cpp] view plaincopyprint?
// 转换为字符串(带格式)   
std::string out = root.toStyledString();  
// 输出无格式json字符串   
Json::FastWriter writer;  
std::string out2 = writer.write(root);  
// 转换为字符串(带格式)
std::string out = root.toStyledString();
// 输出无格式json字符串
Json::FastWriter writer;
std::string out2 = writer.write(root);
二. 使用Boost property_tree解析json
property_tree可以解析xml,json,ini,info等格式的数据,用property_tree解析这几种格式使用方法很相似。

解析json很简单,命名空间为boost::property_tree,reson_json函数将文件流、字符串解析到ptree,write_json将ptree输出为字符串或文件流。其余的都是对ptree的操作。

解析json需要加头文件:

#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/json_parser.hpp>

1. 解析json

解析一段下面的数据:

[html] view plaincopyprint?
{  
  "code": 0,  
  "images":  
  [  
    {  
      "url": "fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg"  
    },  
    {  
      "url": "fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg"  
    }  
  ]  
}  
{
  "code": 0,
  "images":
  [
    {
      "url": "fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg"
    },
    {
      "url": "fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg"
    }
  ]
}
[cpp] view plaincopyprint?
int ParseJson()  
{  
  std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";  
  using namespace boost::property_tree;  
  
  std::stringstream ss(str);  
  ptree pt;  
  try{      
    read_json(ss, pt);  
  }  
  catch(ptree_error & e) {  
    return 1;   
  }  
  
  try{  
    int code = pt.get<int>("code");   // 得到"code"的value   
    ptree image_array = pt.get_child("images");  // get_child得到数组对象   
      
    // 遍历数组   
    BOOST_FOREACH(boost::property_tree::ptree::value_type &v, image_array)  
    {  
      std::stringstream s;  
      write_json(s, v.second);  
      std::string image_item = s.str();  
    }  
  }  
  catch (ptree_error & e)  
  {  
    return 2;  
  }  
  return 0;  
}  
int ParseJson()
{
  std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
  using namespace boost::property_tree;

std::stringstream ss(str);
  ptree pt;
  try{    
    read_json(ss, pt);
  }
  catch(ptree_error & e) {
    return 1; 
  }

try{
    int code = pt.get<int>("code");   // 得到"code"的value
    ptree image_array = pt.get_child("images");  // get_child得到数组对象
    
    // 遍历数组
    BOOST_FOREACH(boost::property_tree::ptree::value_type &v, image_array)
    {
      std::stringstream s;
      write_json(s, v.second);
      std::string image_item = s.str();
    }
  }
  catch (ptree_error & e)
  {
    return 2;
  }
  return 0;
}
2. 构造json

[cpp] view plaincopyprint?
int InsertJson()  
{  
  std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";  
  using namespace boost::property_tree;  
  
  std::stringstream ss(str);  
  ptree pt;  
  try{      
    read_json(ss, pt);  
  }  
  catch(ptree_error & e) {  
    return 1;   
  }  
  
  // 修改/增加一个key-value,key不存在则增加   
  pt.put("upid", "00001");  
  
  // 插入一个数组   
  ptree exif_array;  
  ptree array1, array2, array3;  
  array1.put("Make", "NIKON");  
  array2.put("DateTime", "2011:05:31 06:47:09");  
  array3.put("Software", "Ver.1.01");  
  exif_array.push_back(std::make_pair("", array1));  
  exif_array.push_back(std::make_pair("", array2));  
  exif_array.push_back(std::make_pair("", array3));  
  
//   exif_array.push_back(std::make_pair("Make", "NIKON"));   
//   exif_array.push_back(std::make_pair("DateTime", "2011:05:31 06:47:09"));   
//   exif_array.push_back(std::make_pair("Software", "Ver.1.01"));   
  
  pt.put_child("exifs", exif_array);  
  std::stringstream s2;  
  write_json(s2, pt);  
  std::string outstr = s2.str();  
  
  return 0;  
}  
int InsertJson()
{
  std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
  using namespace boost::property_tree;

std::stringstream ss(str);
  ptree pt;
  try{    
    read_json(ss, pt);
  }
  catch(ptree_error & e) {
    return 1; 
  }

// 修改/增加一个key-value,key不存在则增加
  pt.put("upid", "00001");

// 插入一个数组
  ptree exif_array;
  ptree array1, array2, array3;
  array1.put("Make", "NIKON");
  array2.put("DateTime", "2011:05:31 06:47:09");
  array3.put("Software", "Ver.1.01");
  exif_array.push_back(std::make_pair("", array1));
  exif_array.push_back(std::make_pair("", array2));
  exif_array.push_back(std::make_pair("", array3));

//   exif_array.push_back(std::make_pair("Make", "NIKON"));
//   exif_array.push_back(std::make_pair("DateTime", "2011:05:31 06:47:09"));
//   exif_array.push_back(std::make_pair("Software", "Ver.1.01"));

pt.put_child("exifs", exif_array);
  std::stringstream s2;
  write_json(s2, pt);
  std::string outstr = s2.str();

return 0;
}

三. 两种解析库的使用经验
1. 用boost::property_tree解析字符串遇到"\/"时解析失败,而jsoncpp可以解析成功,要知道'/'前面加一个'\'是JSON标准格式。

2. boost::property_tree的read_json和write_json在多线程中使用会引起崩溃。

针对1,可以在使用boost::property_tree解析前写个函数去掉"\/"中的'\',针对2,在多线程中同步一下可以解决。

我的使用心得:使用boost::property_tree不仅可以解析json,还可以解析xml,info等格式的数据。对于解析json,使用boost::property_tree解析还可以忍受,但解析xml,由于遇到问题太多只能换其它库了。

C++的Json解析库:jsoncpp相关推荐

  1. C++的Json解析库:jsoncpp和boost

    JSON(JavaScript Object Notation)跟xml一样也是一种数据交换格式,了解json请参考其官网http://json.org,本文不再对json做介绍,将重点介绍c++的j ...

  2. [转]C++的Json解析库:jsoncpp和boost

    JSON(JavaScript Object Notation)跟xml一样也是一种数据交换格式,了解json请参考其官网http://json.org,本文不再对json做介绍,将重点介绍c++的j ...

  3. C++的Json解析库:jsoncpp和boost .

    JSON(JavaScript Object Notation)跟xml一样也是一种数据交换格式,了解json请参考其官网http://json.org/,本文不再对json做介绍,将重点介绍c++的 ...

  4. C++ JSON解析之jsoncpp库的使用

    C++ JSON解析之jsoncpp库的使用 1.介绍 轻量级的数据交换格式,采用完全独立于编程语言的文本格式来存储和表示数据,层次结构简洁清晰,易于人阅读和编写,易于机器解析和生成. 2.语法规则 ...

  5. gson解析天气json_几种常用JSON解析库性能比较

    PS:公众号推文时间工作日早晨8点50分,周末下午3点30分,不见不散哈! 作者:飞污熊 xncoding.com/2018/01/09/java/jsons.html 本篇通过JMH来测试一下Jav ...

  6. 深入 Go 中各个高性能 JSON 解析库

    深入 Go 中各个高性能 JSON 解析库 转载请声明出处哦~,本篇文章发布于luozhiyun的博客:https://www.luozhiyun.com/archives/535 其实本来我是没打算 ...

  7. 一个Json解析库的设计和实现

    一个Json解析库的设计和实现 一个Json解析库的设计和实现 设计思路 实现方法 1. 预处理(去除注释) 2. 词法分析 3. 语法分析 4. 树型优化 5. Json树构建 6. 后端处理 整体 ...

  8. Android学习之Json解析库Gson

    接着上一篇Volley,在使用Volley加载好数据之后,我们肯定不能直接使用这个数据,一般获取的数据都会是Json格式,所以自然而然我们要处理下Json,网络上有很多Json解析库,这里我使用Gso ...

  9. iOS开源JSON解析库MJExtension

    iOS中JSON与NSObject互转有两种方式:1.iOS自带类NSJSONSerialization 2.第三方开源库SBJSON.JSONKit.MJExtension.项目中一直用MJExte ...

最新文章

  1. GPIO代码使用流程(伪代码部分示例)
  2. pythonapriori算法特点_Python --深入浅出Apriori关联分析算法(一)
  3. AE中获得datalayer所对应的data的文件路径和文件名
  4. asp.net core中负载均衡场景下http重定向https的问题
  5. ecshop3.0.0注入
  6. 12.FreerRTOS学习笔记-链表的使用
  7. 全球顶级开源大神们现身 COSCon‘20
  8. swift 项目_谷歌为何要养苹果的亲儿子Swift?原来意在可微分编程
  9. Microsoft buildup new website for open-source project
  10. 农林牧渔行业S2B2C系统网站提升品牌知名度,提升盈利水平
  11. linux就该这么学【进程地址空间】
  12. java 微信请求超时_java – SQL服务器“超出锁定请求超时时间”..再次
  13. STM32 高级定时TIM 死区时间计算--C语言实现
  14. 计算机工作原理与系统组成?,计算机工作原理及系统组成
  15. 《SiamMask:Fast Online Object Tracking and Segmentation:A Unifying Approach》论文笔记
  16. 计算机应用与需求相结合,计算机应用人才培养与企业需求的有效对接
  17. Interceptor for {http://admin.ws…thrown exception, unwinding now java.lang.NullPointerException
  18. 混沌工程之ChaosBlade(一):建立混沌工程思想
  19. php酒店入住时间怎么算,宾馆早上6点入住怎么算?正规宾馆上午八点开房应该到什么时间算一天...
  20. 微信内置浏览器清理缓存方法

热门文章

  1. (二)开集识别学习 open-set recognition(OSR)
  2. 赛扬处理器_海尔逸15笔记本仅2299元,其酷睿10代赛扬处理器是什么情况?
  3. 【独家专访】谭龙喊话同行:几个月后 ETCP将不可超越
  4. 高效学习-笔记-左耳听风
  5. vue: Invalid prop: type check failed for prop “action“. Expected String with valu
  6. ubuntu10.04换官方源
  7. wordpress 主题 neutral 下载
  8. Fluent速度、压力入口边界条件的设定
  9. 用负片制作中途曝光作品
  10. Citrix 服务器虚拟化之二十五 桌面虚拟化之XenDesktop高可用性