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
- int ParseJsonFromString()
- {
- constchar* 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文件内容:
- {
- "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" } ] } ] }
解析代码:
- int ParseJsonFromFile(constchar* 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
- 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
- // 转换为字符串(带格式)
- 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
解析一段下面的数据:
- {
- "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" } ] }
- int ParseJson()
- {
- std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
- usingnamespace 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
- int InsertJson()
- {
- std::string str = "{\"code\":0,\"images\":[{\"url\":\"fmn057/20111221/1130/head_kJoO_05d9000251de125c.jpg\"},{\"url\":\"fmn057/20111221/1130/original_kJoO_05d9000251de125c.jpg\"}]}";
- usingnamespace 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就是这样一款优秀的C++ JSON解析器。它不依赖于Boost库,这使得它的集成更加简便,也减少了项目对额外库的依赖。 jsoncpp库提供了对JSON数据的解析、生成...
JSONCPP 是一个开源库,专为C++设计,用于处理JSON(JavaScript Object Notation)数据格式。JSON是一种轻量级的数据交换格式,被广泛应用于Web服务和应用程序之间的数据传输。在JSONCPP中,可能会遇到写入文件时...
在C++编程中,为了方便处理JSON数据,开发者通常会使用JSON解析库,比如JsonCpp。JsonCpp是一个开源的C++库,它提供了读取、写入和操作JSON数据的能力。 JsonCpp的主要功能包括: 1. JSON解析:JsonCpp可以将JSON...
JsonCpp库提供了读取、写入和操作JSON数据的功能,使得在C++程序中处理JSON变得简单。 JsonCpp的源码编译在不同环境中可能会遇到一些问题,特别是在较旧的编译器如Visual Studio 2008和2010中。在这个案例中,用户...
JSONCPP是一个开源库,专为解析、生成和操作JSON(JavaScript Object Notation)数据而设计。JSON是一种轻量级的数据交换格式,广泛应用于Web服务和跨平台通信。JSONCPP库基于C++,提供了方便的API,使得在C++程序中...
JSONCPP库提供了C++语言对JSON的支持,允许开发者方便地解析、生成和操作JSON格式的数据。 标题“jsoncpp-0.10.6-master.zip”表明这是JSONCPP库的一个特定版本,版本号为0.10.6,并且可能包含了源代码和相关构建...
3. **Jsoncpp库**:Jsoncpp是一个用于解析和生成JSON数据的C++库。在WebSocket应用中,JSON经常用于数据序列化和反序列化,便于数据交换。因此,jsoncpp库可能用于将C++对象转换为JSON格式,或者将接收到的JSON消息...
获取到的响应通常是以JSON格式返回的,然后使用选定的JSON库解析这个字符串。 2. **数据模型**:根据豆瓣书籍API的结构,创建对应的C++数据结构,如`Book`类,包含书籍的ID、标题、作者、评分等属性。这些数据结构...
综上所述,C++通过VS开发获取网络天气信息涉及了HTTP协议的理解、选择合适的库(libcurl或Winsock API)发送请求、接收和解析服务器响应,以及可能的JSON解析。确保在实际项目中正确处理错误情况,确保良好的用户...
JSONCPP是JSON(JavaScript Object Notation)的一种C++实现,它提供了一套库来解析、生成、操作JSON格式的数据。JSON是一种轻量级的数据交换格式,广泛用于Web服务和应用程序之间的数据传输。以下是在Linux CentOS ...
JSONCPP是一个开源的C++库,专门用于解析和生成JSON(JavaScript Object Notation)格式的数据。JSON是一种轻量级的数据交换格式,广泛应用于Web服务和跨平台通信,因为它的结构清晰,易于人类阅读和编写,同时也...
rocketMQ 官方提供的C++ sdk工程源码...似乎官方对C++sdk不太重视,可能和rocketMQ 的关注度低有关。在这里把自己编译好的动态库放上来供使用者直接下载。还有相关依赖库也附上。 jsoncpp-0.10.6 编译环境 vs2015 x64
总的来说,"Class_to_json.zip"项目提供了一个使用C++、Boost和jsoncpp将类对象转换为JSON的示例。它展示了如何利用元编程和反射来访问类的成员,以及如何通过jsoncpp将这些数据转换为JSON格式。这样的技术在许多...
4. **tolua_jsoncpp**:JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,jsoncpp是C++的JSON解析库。通过tolua_jsoncpp,可以在Lua脚本中方便地处理JSON数据,进行序列化和反序列化操作。 5. **...
C++有多个库如jsoncpp或pugixml用于解析和生成这些格式。 7. **异常处理**:网络编程中,错误处理和异常处理至关重要,确保程序的健壮性。C++的try-catch机制可以用来捕获和处理可能出现的异常情况。 8. **跨平台...
7. **Jsoncpp or Rapidjson**: 这两个库用于JSON数据的解析和序列化,便于在客户端和服务器之间交换数据。 8. **Zlib**: 用于数据压缩和解压缩,如果MUD游戏需要传输大量数据,Zlib可以帮助减小网络负担。 9. **...
由于JSON被广泛用于数据交换格式,所以项目中还加入了`jsoncpp`这个库,它是一个JSON的解析器和生成器,用于将C++对象转换为JSON字符串,或者将接收到的JSON数据解析成C++对象。 JSON(JavaScript Object Notation...
jsoncpp是一个C++库,支持JSON的解析和序列化,这对于处理API请求和响应非常有用。 6. **cpp-httplib的使用和原理**:cpp-httplib是一个轻量级的HTTP服务器和客户端库,适用于C++。它可以帮助我们快速实现Web服务器...
此外,C++还能通过API接口与金融机构交互,获取股票历史数据和其他金融信息,libcurl库用于发送HTTP请求,jsoncpp用于解析返回的JSON数据。 2. **策略编写**:C++拥有丰富的数学计算库,如Eigen用于矩阵运算和特征...
3. **数据序列化和解析**:消息在传输前需要被编码为字节流,可能使用JSON、XML或自定义格式。C++标准库没有内置的序列化支持,所以开发者可能使用第三方库如protobuf或jsoncpp。 4. **用户界面**:为了提供友好的...