• C++ 解析Json——jsoncpp(转)


     原文:https://www.cnblogs.com/liaocheng/p/4243731.html 

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,和xml类似,本文主要对VS2008中使用Jsoncpp解析json的方法做一下记录。
    Jsoncpp是个跨平台的开源库,下载地址:http://sourceforge.net/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.0jsoncpp-src-0.5.0includejson和jsoncpp-src-0.5.0jsoncpp-src-0.5.0srclib_json目录里的文件包含到VS工程中,在VS工程的属性C/C++下General中Additional Include Directories包含头文件目录.jsoncpp-src-0.5.0include。在使用的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 
    ", 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; i<10; 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<arrayObj.size(); 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 
    ",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 &current = 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();
            }
        }

  • 相关阅读:
    linux从入门到精通(五)(更改文件的所有者,所属组)
    【洛谷】4180:【模板】严格次小生成树[BJWC2010]【链剖】【线段树维护最大、严格次大值】
    CF980E The Number Games【树链剖分/线段树】
    【POJ】1419:Graph Coloring【普通图最大点独立集】【最大团】
    【HDU】2866:Special Prime【数论】
    【BZOJ】1415: [Noi2005]聪聪和可可【期望】【最短路】【记忆化搜索】
    【BZOJ】2561: 最小生成树【网络流】【最小割】
    【洛谷】3966:[TJOI2013]单词【AC自动机】【fail树】
    【10.17校内测试】【二进制数位DP】【博弈论/预处理】【玄学(?)DP】
    【洛谷】2602: [ZJOI2010]数字计数【数位DP】
  • 原文地址:https://www.cnblogs.com/z3286586/p/10416351.html
Copyright © 2020-2023  润新知