分享

C 解析Json——jsoncpp

 西北望msm66g9f 2018-07-07

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

 


方法一:使用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\json和jsoncpp-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();  }

 4.序列化json字符串

先构建一个Json对象,此Json对象中含有数组,然后把Json对象序列化成字符串,代码如下:    Json::Value root;    Json::Value arrayObj;    Json::Value item;    for (int i=0; i10; i++)    {      item['key'] = i;      arrayObj.append(item);    }    root['key1'] = “value1″;    root['key2'] = “value2″;    root['array'] = arrayObj;    root.toStyledString();    std::string out = root.toStyledString();    std::cout <>out std::endl;

 

5.反序列化json

比如一个Json对象的字符串序列如下,其中”array”:[...]表示Json对象中的数组:{“key1″:”value1″,”array”:[{'key2':'value2'},{'key2':'value3'},{'key2':'value4'}]},那怎么分别取到key1和key2的值呢,代码如下所示:    std::string strValue = “{\”key1\”:\”value1\”,\”array\”:[{\'key2\':\'value2\'},{\'key2\':\'value3\'},{\'key2\':\'value4\'}]}”;    Json::Reader reader;    Json::Value value;    if (reader.parse(strValue, value))    {      std::string out = value['key1'].asString();      std::cout <>out std::endl;      const Json::Value arrayObj = value['array'];      for (int i=0; i)      {        out = arrayObj[i]['key2'].asString();        std::cout <>out;        if (i != arrayObj.size() – 1 )          std::cout std::endl;      }    }

6.删除json子对象

   std::string strContent = '{\'key\':\'1\',\'name\':\'test\'}';    Json::Reader reader;    Json::Value value;    if (reader.parse(strContent, value))    {        Json::Value root=value;        root.removeMember('key');        printf('%s \n',root.toStyledString().c_str());
  }

 7. 利用jsoncpp将json字符串转换为Vector

在API测试过程中经常会遇到传入参数为复杂类型,一般情况下在python下,习惯用字典来表示复杂类型。但是c++对字符串的处理是比较弱智的,一般c++里边会用vector来存储复杂类型,那么就存在转换的问题,下面小段代码记录了将字符串转换为Vector的过程

待转换的字符串如下:

const char * jsongroupinfo='[{/'groupId/' :946838524,/'groupname/' :/'bababa/', /'mask/':1,/'parentid/':946755072}]'; Json::Reader reader;Json::Value json_object;if (!reader.parse(jsongroupinfo, json_object))  return 'parse jsonstr error';SUserChggroup sucg;VECTOR< suserchggroup=""> m_groupInfo;for(int i = 0; i < json_object.size();="" i="">){  Json::Value ¤t = json_object[i];  sucg.m_groupId = current['groupId'].asInt();  sucg.m_groupName = current['groupname'].asString();  sucg.m_mask = current['mask'].asInt();  sucg.m_parentId = current['parentid'].asInt();  m_groupInfo.push_back(sucg);}

 

 

简而言之,就是把它变成解析成一个个对象,再将对象存储到vector中。

 

ps:在使用vs2005调用vs2010编译的dll时候,出现string的内存错误,在不同版本的string的不能相互传递,用const char *比较好,相关代码修改:

   std::string strRtn = '{'success':true,'user':{'id':6,'username':'wq','type':'admin','membership':{'type':'paid','expiredAt':'2019-07-28T16:00:00.000Z'}},'token':'8de57200-3235-11e8-83f1-9739d2f0386f'}';    std::string strToken;    Json::Reader reader;                                    //解析json用Json::Reader Json::Value value; //可以代表任意类型 if (reader.parse(strRtn.c_str(),strRtn.c_str()+strRtn.size(), value))      {          if (value['success'].asBool())                {            strToken = value['token'].asCString();        }    }


    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多