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

[cpp]  view plain copy
  1. int ParseJsonFromString()  
  2. {  
  3.   const char* str = "{\"uploadid\": \"UP000000\",\"code\": 100,\"msg\": \"\",\"files\": \"\"}";  
  4.   
  5.   Json::Reader reader;  
  6.   Json::Value root;  
  7.   if (reader.parse(str, root))  // reader将Json字符串解析到root,root将包含Json里所有子元素  
  8.   {  
  9.     std::string upload_id = root["uploadid"].asString();  // 访问节点,upload_id = "UP000000"  
  10.     int code = root["code"].asInt();    // 访问节点,code = 100  
  11.   }  
  12.   return 0;  
  13. }  

2. 从文件解析json

json文件内容:

[cpp]  view plain copy
  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. }  

 解析代码:

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

 3. 在json结构中插入json

[cpp]  view plain copy
  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中  

 4. 输出json

[cpp]  view plain copy
  1. // 转换为字符串(带格式)  
  2. std::string out = root.toStyledString();  
  3. // 输出无格式json字符串  
  4. Json::FastWriter writer;  
  5. 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 plain copy
  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. }  
[cpp]  view plain copy
  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.   using namespace boost::property_tree;  
  5.   
  6.   std::stringstream ss(str);  
  7.   ptree pt;  
  8.   try{      
  9.     read_json(ss, pt);  
  10.   }  
  11.   catch(ptree_error & e) {  
  12.     return 1;   
  13.   }  
  14.   
  15.   try{  
  16.     int code = pt.get<int>("code");   // 得到"code"的value  
  17.     ptree image_array = pt.get_child("images");  // get_child得到数组对象  
  18.       
  19.     // 遍历数组  
  20.     BOOST_FOREACH(boost::property_tree::ptree::value_type &v, image_array)  
  21.     {  
  22.       std::stringstream s;  
  23.       write_json(s, v.second);  
  24.       std::string image_item = s.str();  
  25.     }  
  26.   }  
  27.   catch (ptree_error & e)  
  28.   {  
  29.     return 2;  
  30.   }  
  31.   return 0;  
  32. }  

2. 构造json

[cpp]  view plain copy
  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.   using namespace boost::property_tree;  
  5.   
  6.   std::stringstream ss(str);  
  7.   ptree pt;  
  8.   try{      
  9.     read_json(ss, pt);  
  10.   }  
  11.   catch(ptree_error & e) {  
  12.     return 1;   
  13.   }  
  14.   
  15.   // 修改/增加一个key-value,key不存在则增加  
  16.   pt.put("upid""00001");  
  17.   
  18.   // 插入一个数组  
  19.   ptree exif_array;  
  20.   ptree array1, array2, array3;  
  21.   array1.put("Make""NIKON");  
  22.   array2.put("DateTime""2011:05:31 06:47:09");  
  23.   array3.put("Software""Ver.1.01");  
  24.   exif_array.push_back(std::make_pair("", array1));  
  25.   exif_array.push_back(std::make_pair("", array2));  
  26.   exif_array.push_back(std::make_pair("", array3));  
  27.   
  28. //   exif_array.push_back(std::make_pair("Make", "NIKON"));  
  29. //   exif_array.push_back(std::make_pair("DateTime", "2011:05:31 06:47:09"));  
  30. //   exif_array.push_back(std::make_pair("Software", "Ver.1.01"));  
  31.   
  32.   pt.put_child("exifs", exif_array);  
  33.   std::stringstream s2;  
  34.   write_json(s2, pt);  
  35.   std::string outstr = s2.str();  
  36.   
  37.   return 0;  
  38. }  

 

三. 两种解析库的使用经验

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,由于遇到问题太多只能换其它库了。

============================================================================================================

JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,和xml类似,本文主要对VS2008中使用Jsoncpp解析json的方法做一下记录。
Jsoncpp是个跨平台的开源库,下载地址:http://sourceforge.net/projects/jsoncpp/,我下载的是v0.5.0,压缩包大约104K。

方法一:使用Jsoncpp生成的lib文件
      解压上面下载的Jsoncpp文件,在jsoncpp-src-0.5.0/makefiles/vs71目录里找到jsoncpp.sln,用VS2008版本编译,默认生成静态链接库。 在工程中引用,只需要包含include/json下的头文件及生成的.lib文件即可。
      如何包含lib文件:在.cpp文件中#pragma comment(lib."json_vc71_libmt.lib"),在工程属性中Linker下Input中Additional Dependencies写入lib文件名字(Release下为json_vc71_libmt.lib,Debug为json_vc71_libmtd.lib

注意:Jsoncpp的lib工程编译选项要和VS工程中的编译选项保持一致。如lib文件工程编译选项为MT(或MTd),VS工程中也要选择MT(或MTd),否则会出现编译错误问题,debug和release下生成的lib文件名字不同,注意不要看错了,当成一个文件来使用(我就犯了这个错误)。

方法二:使用Jsoncpp包中的.cpp.h文件
      解压上面下载的Jsoncpp文件,把jsoncpp-src-0.5.0文件拷贝到工程目录下,jsoncpp-src-0.5.0\jsoncpp-src-0.5.0\include\jsonjsoncpp-src-0.5.0\jsoncpp-src-0.5.0\src\lib_json目录里的文件包含到VS工程中,在VS工程的属性C/C++下General中Additional Include Directories包含头文件目录.\jsoncpp-src-0.5.0\include。在使用的cpp文件中包含json头文件即可,如:#include "json/json.h"。将json_reader.cpp、json_value.cpp和json_writer.cpp三个文件的Precompiled Header属性设置为Not Using Precompiled Headers否则编译会出现错误。

jsoncpp 使用详解

jsoncpp 主要包含三种类型的 class:Value、Reader、Writer。jsoncpp 中所有对象、类名都在 namespace Json 中,包含 json.h 即可。

Json::Value 只能处理 ANSI 类型的字符串,如果 C++ 程序是用 Unicode 编码的,最好加一个 Adapt 类来适配。


下面是从网上找的代码示例:
1. 从字符串解析json
     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 
    }  
2. 从文件解析json
int  ReadJsonFromFile( const  char * filename)  
{  
    Json::Reader reader; //  解析json用Json::Reader   
    Json::Value root;  //  Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array         

    std::ifstream  is ;  
     is .open (filename, std::ios::binary );    
     if  (reader.parse( is , root, FALSE))  
    {  
        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();  
        
        code = root. get ("uploadid", "null").asString(); //  访问节点,Return the member named key if it exist, defaultValue otherwise.    

         int  file_size = root["files"].size();   //  得到"files"的数组个数  
         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(); 
                printf("type : %s, url : %s \n", type.c_str(), url.c_str());
            }  
        }  
    }  
     is .close();  

     return  0;  
3. 向文件中插入json
void  WriteJsonData( const  char * filename)
{
    Json::Reader reader;  
    Json::Value root;  //  Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array        

    std::ifstream  is ;  
     is .open (filename, std::ios::binary );    
     if  (reader.parse( is , root))  
    {  
        Json::Value arrayObj;    //  构建对象  
        Json::Value new_item, new_item1;  
        new_item["date"] = "2011-11-11";  
        new_item1["time"] = "11:11:11";  
        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中 
        std:: string  out  = root.toStyledString();  
         //  输出无格式json字符串  
        Json::FastWriter writer;  
        std:: string  strWrite = writer.write(root);
        std::ofstream ofs;
        ofs.open("test_write.json");
        ofs << strWrite;
        ofs.close();
    }  

     is .close();  
}

下面是我写的 使用 Jsoncpp生成lib文件的工程示例:Jsoncpp测试代码,使用VS2008编译通过。

// TestJsoncppCode.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include "TestJsoncppCode.h"
#include "include/json/json.h"
#include <fstream>
#include <string>

#pragma comment(lib."json_vc71_libmtd.lib")

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

// The one and only application object

CWinApp theApp;
using namespace std;
CString GetWorkDir() 
{  
	char pFileName[MAX_PATH]; 
	int nPos = GetCurrentDirectory( MAX_PATH, pFileName); 

	CString csFullPath(pFileName);  
	if( nPos < 0 ) 
		return CString(""); 
	else 
		return csFullPath; 
}

void WriteJsonData(const char* filename)
{
	Json::Reader reader;  
	Json::Value root; // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array...       
	std::ifstream is;  
	is.open (filename, std::ios::binary );    
	if (reader.parse(is, root))  
	{  
		Json::Value arrayObj;   // 构建对象  
		Json::Value new_item, new_item1;  
		new_item["date"] = "2011-11-11";  
		new_item1["time"] = "11:11:11";  
		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中 
		std::string out = root.toStyledString();  
		// 输出无格式json字符串  
		Json::FastWriter writer;  
		std::string strWrite = writer.write(root);
		std::ofstream ofs;
		ofs.open("test_write.json");
		ofs << strWrite;
		ofs.close();
	}  

	is.close();  
}

int ReadJsonFromFile(const char* filename)  
{  
	Json::Reader reader;// 解析json用Json::Reader   
	Json::Value root; // Json::Value是一种很重要的类型,可以代表任意类型。如int, string, object, array...         
	std::ifstream is;  
	is.open (filename, std::ios::binary );    
	if (reader.parse(is, root, FALSE))  
	{  
		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();  
		
		code = root.get("uploadid", "null").asString();// 访问节点,Return the member named key if it exist, defaultValue otherwise.    

		int file_size = root["files"].size();  // 得到"files"的数组个数  
		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(); 
				printf("type : %s, url : %s \n", type.c_str(), url.c_str());
			}  
		}  
	}  
	is.close();  
	return 0;  
}  

int _tmain(int argc, TCHAR* argv[], TCHAR* envp[])
{
	int nRetCode = 0;

	// initialize MFC and print and error on failure
	if (!AfxWinInit(::GetModuleHandle(NULL), NULL, ::GetCommandLine(), 0))
	{
		// TODO: change error code to suit your needs
		_tprintf(_T("Fatal Error: MFC initialization failed\n"));
		nRetCode = 1;
	}
	else
	{
		// TODO: code your application's behavior here.
	}

	//1.从字符串解析json
	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里所有子元素  
	{  
		printf("--------------从字符串读取JSON---------------\n");
		std::string upload_id = root["uploadid"].asString();  // 访问节点,upload_id = "UP000000"  
		int code = root["code"].asInt();                      // 访问节点,code = 100 

		printf("upload_id : %s\ncode : %d \n", upload_id.c_str(), code);
	}  

	//2.从文件解析json
	CString sTempPath = GetWorkDir() + "\\test_json.json";
	if (!CFileFind().FindFile(sTempPath))
	{
		printf("Not Find JSON File : %s", sTempPath);
	}
	else
	{
		printf("--------------从文件读取JSON---------------\n");
		ReadJsonFromFile(sTempPath);
	}

	//3.向文件写入json
	WriteJsonData(sTempPath);

	system("pause");
	return nRetCode;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值