`

C++的Json解析库:jsoncpp和boost (转)

 
阅读更多

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

  1. int ParseJsonFromString()
  2. {
  3. constchar* str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}";
  4. Json::Reader reader;
  5. Json::Value root;
  6. if (reader.parse(str, root)) // reader将Json字符串解析到root,root将包含Json里所有子元素
  7. {
  8. std::string upload_id = root["uploadid"].asString(); // 访问节点,upload_id = "UP000000"
  9. int code = root["code"].asInt(); // 访问节点,code = 100
  10. }
  11. return 0;
  12. }
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文件内容:

  1. {
  2. "uploadid": "UP000000",
  3. "code": "0",
  4. "msg": "",
  5. "files":
  6. [
  7. {
  8. "code": "0",
  9. "msg": "",
  10. "filename": "1D_16-35_1.jpg",
  11. "filesize": "196690",
  12. "width": "1024",
  13. "height": "682",
  14. "images":
  15. [
  16. {
  17. "url": "fmn061/20111118",
  18. "type": "large",
  19. "width": "720",
  20. "height": "479"
  21. },
  22. {
  23. "url": "fmn061/20111118",
  24. "type": "main",
  25. "width": "200",
  26. "height": "133"
  27. }
  28. ]
  29. }
  30. ]
  31. }
{
    "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"
                }
            ]
        }
    ]
}

解析代码:

  1. int ParseJsonFromFile(constchar* filename)
  2. {
  3. // 解析json用Json::Reader
  4. Json::Reader reader;
  5. // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array...
  6. Json::Value root;
  7. std::ifstream is;
  8. is.open (filename, std::ios::binary );
  9. if (reader.parse(is, root))
  10. {
  11. std::string code;
  12. if (!root["files"].isNull()) // 访问节点,Access an object value by name, create a null member if it does not exist.
  13. code = root["uploadid"].asString();
  14. // 访问节点,Return the member named key if it exist, defaultValue otherwise.
  15. code = root.get("uploadid", "null").asString();
  16. // 得到"files"的数组个数
  17. int file_size = root["files"].size();
  18. // 遍历数组
  19. for(int i = 0; i < file_size; ++i)
  20. {
  21. Json::Value val_image = root["files"][i]["images"];
  22. int image_size = val_image.size();
  23. for(int j = 0; j < image_size; ++j)
  24. {
  25. std::string type = val_image[j]["type"].asString();
  26. std::string url = val_image[j]["url"].asString();
  27. }
  28. }
  29. }
  30. is.close();
  31. return 0;
  32. }
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

  1. Json::Value arrayObj; // 构建对象
  2. Json::Value new_item, new_item1;
  3. new_item["date"] = "2011-12-28";
  4. new_item1["time"] = "22:30:36";
  5. arrayObj.append(new_item); // 插入数组成员
  6. arrayObj.append(new_item1); // 插入数组成员
  7. int file_size = root["files"].size();
  8. for(int i = 0; i < file_size; ++i)
  9. 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

  1. // 转换为字符串(带格式)
  2. std::string out = root.toStyledString();
  3. // 输出无格式json字符串
  4. Json::FastWriter writer;
  5. 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

解析一段下面的数据:

  1. {
  2. "code": 0,
  3. "images":
  4. [
  5. {
  6. "url": "fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg"
  7. },
  8. {
  9. "url": "fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg"
  10. }
  11. ]
  12. }
{
  "code": 0,
  "images":
  [
    {
      "url": "fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg"
    },
    {
      "url": "fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg"
    }
  ]
}
  1. int ParseJson()
  2. {
  3. std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
  4. usingnamespace boost::property_tree;
  5. std::stringstream ss(str);
  6. ptree pt;
  7. try{
  8. read_json(ss, pt);
  9. }
  10. catch(ptree_error & e) {
  11. return 1;
  12. }
  13. try{
  14. int code = pt.get<int>("code"); // 得到"code"的value
  15. ptree image_array = pt.get_child("images"); // get_child得到数组对象
  16. // 遍历数组
  17. BOOST_FOREACH(boost::property_tree::ptree::value_type &v, image_array)
  18. {
  19. std::stringstream s;
  20. write_json(s, v.second);
  21. std::string image_item = s.str();
  22. }
  23. }
  24. catch (ptree_error & e)
  25. {
  26. return 2;
  27. }
  28. return 0;
  29. }
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

  1. int InsertJson()
  2. {
  3. std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
  4. usingnamespace boost::property_tree;
  5. std::stringstream ss(str);
  6. ptree pt;
  7. try{
  8. read_json(ss, pt);
  9. }
  10. catch(ptree_error & e) {
  11. return 1;
  12. }
  13. // 修改/增加一个key-value,key不存在则增加
  14. pt.put("upid", "00001");
  15. // 插入一个数组
  16. ptree exif_array;
  17. ptree array1, array2, array3;
  18. array1.put("Make", "NIKON");
  19. array2.put("DateTime", "2011:05:31 06:47:09");
  20. array3.put("Software", "Ver.1.01");
  21. exif_array.push_back(std::make_pair("", array1));
  22. exif_array.push_back(std::make_pair("", array2));
  23. exif_array.push_back(std::make_pair("", array3));
  24. // exif_array.push_back(std::make_pair("Make", "NIKON"));
  25. // exif_array.push_back(std::make_pair("DateTime", "2011:05:31 06:47:09"));
  26. // exif_array.push_back(std::make_pair("Software", "Ver.1.01"));
  27. pt.put_child("exifs", exif_array);
  28. std::stringstream s2;
  29. write_json(s2, pt);
  30. std::string outstr = s2.str();
  31. return 0;
  32. }
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,由于遇到问题太多只能换其它库了。

 

from:http://blog.csdn.net/hzyong_c/article/details/7163589

分享到:
评论

相关推荐

    jsoncpp (很好用的C++ Json解析器)

    在C++编程中,处理JSON数据通常需要借助第三方库,而jsoncpp就是这样一款优秀的C++ JSON解析器。它不依赖于Boost库,这使得它的集成更加简便,也减少了项目对额外库的依赖。 jsoncpp库提供了对JSON数据的解析、生成...

    jsoncpp 写入乱码解决版本

    JSONCPP 是一个开源库,专为C++设计,用于处理JSON(JavaScript Object Notation)数据格式。JSON是一种轻量级的数据交换格式,被广泛应用于Web服务和应用程序之间的数据传输。在JSONCPP中,可能会遇到写入文件时...

    jsoncpp-src-0.5.0.rar

    在C++编程中,为了方便处理JSON数据,开发者通常会使用JSON解析库,比如JsonCpp。JsonCpp是一个开源的C++库,它提供了读取、写入和操作JSON数据的能力。 JsonCpp的主要功能包括: 1. JSON解析:JsonCpp可以将JSON...

    JsonCpp源码VS2008/VS2010编译成功

    JsonCpp库提供了读取、写入和操作JSON数据的功能,使得在C++程序中处理JSON变得简单。 JsonCpp的源码编译在不同环境中可能会遇到一些问题,特别是在较旧的编译器如Visual Studio 2008和2010中。在这个案例中,用户...

    jsoncpp-master-based-on-1.8.4.zip

    JSONCPP是一个开源库,专为解析、生成和操作JSON(JavaScript Object Notation)数据而设计。JSON是一种轻量级的数据交换格式,广泛应用于Web服务和跨平台通信。JSONCPP库基于C++,提供了方便的API,使得在C++程序中...

    jsoncpp-0.10.6-master.zip

    JSONCPP库提供了C++语言对JSON的支持,允许开发者方便地解析、生成和操作JSON格式的数据。 标题“jsoncpp-0.10.6-master.zip”表明这是JSONCPP库的一个特定版本,版本号为0.10.6,并且可能包含了源代码和相关构建...

    c++使用websocket代码Demo.可直接编译

    3. **Jsoncpp库**:Jsoncpp是一个用于解析和生成JSON数据的C++库。在WebSocket应用中,JSON经常用于数据序列化和反序列化,便于数据交换。因此,jsoncpp库可能用于将C++对象转换为JSON格式,或者将接收到的JSON消息...

    C+JSON豆瓣书籍源码

    获取到的响应通常是以JSON格式返回的,然后使用选定的JSON库解析这个字符串。 2. **数据模型**:根据豆瓣书籍API的结构,创建对应的C++数据结构,如`Book`类,包含书籍的ID、标题、作者、评分等属性。这些数据结构...

    C++开发之路:网络获取天气信息(VS开发)

    综上所述,C++通过VS开发获取网络天气信息涉及了HTTP协议的理解、选择合适的库(libcurl或Winsock API)发送请求、接收和解析服务器响应,以及可能的JSON解析。确保在实际项目中正确处理错误情况,确保良好的用户...

    jsoncpp安装步骤

    JSONCPP是JSON(JavaScript Object Notation)的一种C++实现,它提供了一套库来解析、生成、操作JSON格式的数据。JSON是一种轻量级的数据交换格式,广泛用于Web服务和应用程序之间的数据传输。以下是在Linux CentOS ...

    JSONCPP.zip

    JSONCPP是一个开源的C++库,专门用于解析和生成JSON(JavaScript Object Notation)格式的数据。JSON是一种轻量级的数据交换格式,广泛应用于Web服务和跨平台通信,因为它的结构清晰,易于人类阅读和编写,同时也...

    jsoncpp-0.10.6.rar

    rocketMQ 官方提供的C++ sdk工程源码...似乎官方对C++sdk不太重视,可能和rocketMQ 的关注度低有关。在这里把自己编译好的动态库放上来供使用者直接下载。还有相关依赖库也附上。 jsoncpp-0.10.6 编译环境 vs2015 x64

    Class_to_json.zip

    总的来说,"Class_to_json.zip"项目提供了一个使用C++、Boost和jsoncpp将类对象转换为JSON的示例。它展示了如何利用元编程和反射来访问类的成员,以及如何通过jsoncpp将这些数据转换为JSON格式。这样的技术在许多...

    cpp-为xLua集成几个常用库方便使用

    4. **tolua_jsoncpp**:JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,jsoncpp是C++的JSON解析库。通过tolua_jsoncpp,可以在Lua脚本中方便地处理JSON数据,进行序列化和反序列化操作。 5. **...

    一个实现BS结构的c++程序

    C++有多个库如jsoncpp或pugixml用于解析和生成这些格式。 7. **异常处理**:网络编程中,错误处理和异常处理至关重要,确保程序的健壮性。C++的try-catch机制可以用来捕获和处理可能出现的异常情况。 8. **跨平台...

    Libraries_C++_

    7. **Jsoncpp or Rapidjson**: 这两个库用于JSON数据的解析和序列化,便于在客户端和服务器之间交换数据。 8. **Zlib**: 用于数据压缩和解压缩,如果MUD游戏需要传输大量数据,Zlib可以帮助减小网络负担。 9. **...

    websocket,cpp,html5,json

    由于JSON被广泛用于数据交换格式,所以项目中还加入了`jsoncpp`这个库,它是一个JSON的解析器和生成器,用于将C++对象转换为JSON字符串,或者将接收到的JSON数据解析成C++对象。 JSON(JavaScript Object Notation...

    c++博客系统1

    jsoncpp是一个C++库,支持JSON的解析和序列化,这对于处理API请求和响应非常有用。 6. **cpp-httplib的使用和原理**:cpp-httplib是一个轻量级的HTTP服务器和客户端库,适用于C++。它可以帮助我们快速实现Web服务器...

    C++在量化交易方面的应用

    此外,C++还能通过API接口与金融机构交互,获取股票历史数据和其他金融信息,libcurl库用于发送HTTP请求,jsoncpp用于解析返回的JSON数据。 2. **策略编写**:C++拥有丰富的数学计算库,如Eigen用于矩阵运算和特征...

    关于C++聊天的工具

    3. **数据序列化和解析**:消息在传输前需要被编码为字节流,可能使用JSON、XML或自定义格式。C++标准库没有内置的序列化支持,所以开发者可能使用第三方库如protobuf或jsoncpp。 4. **用户界面**:为了提供友好的...

Global site tag (gtag.js) - Google Analytics