jsoncpp初探

1.jsoncpp简介

首先说一下JSON。JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。它基于ECMAScript的一个子集。 JSON采用完全独立于语言的文本格式,这些特性使JSON成为理想的数据交换语言,易于人阅读和编写,同时也易于机器解析和生成,一般用于网络传输。

jsoncpp就是用于生成和解析json的C++开源代码库。 官网见:here,源代码托管在github:here

除了jsoncpp,可供我们选择的第三方开源的用于解析和生成json的C++库还有很多。比如SimpleJSONjson-spirit腾讯的RapidJSON,当然还有C语言的cJSONlibJSON。这里之所以介绍jsoncpp是因为我周边的人用的比较多,相比腾讯的RapidJSON,本来是想用RapidJSON,但是到github上一看,被那纷杂的头文件给吓到了,虽然jsoncpp的头文件也很多,但是相对来说还是少很多。还有一点,jsoncpp用的人多,网上解决问题的资源也就相对多一点。

这里贴一张RapidJSON的作者Milo Yip对网上开源的json库的测评图,测试环境Corei5-3330S@2.70GHz_mac64_clang6.1_1

解析json字符串的时间:

具体参见github Milo Yip native-json benchmark

1.1jsoncpp的内容

(1)JsonCpp主要包含三种类型:Value、Reader和Writer;

Json::Value是jsoncpp 中最基本、最重要的类,用于表示各种类型的对象,jsoncpp 支持的对象类型可见 Json::ValueType 枚举值。

Json::Reader是用于读取的,说的确切点,是用于将字符串转换为 Json::Value 对象的。

Json::Writer类是一个纯虚类,并不能直接使用。在此我们使用 Json::Writer 的子类:Json::FastWriter、Json::StyledWrite、Json::StyledStreamWriter。

(2)Jsoncpp中所有对象、类名都在namespace json中,包含json.h即可

2.jsoncpp下载与编译

第一步:请到github上下载jsoncpp源码,点击这里

github上托管的是jsoncpp整个项目,包括了很多乱七八糟的文件,比如说明文件README.md,示例代码,构建文件cmake.txt,还有各种乱起八糟的脚本文件,不得不吐槽一下,真的很乱。我们需要的仅仅只是其源码,在说明文件中还没有显明的指出源码的所在目录,这点做的不好啊,看了很多其它的开源项目,不仅仅是jsoncpp没有说明,很多都没有说明。还要自己去摸索,筛选自己真真需要的那几个源码文件,真的很令人头痛。

好了,我们需要的源码文件有目录/include/json/ 下的所有头文件,还有目录/src/lib_json/ 下的源文件。这里我又要吐槽一下(可能我有简洁强迫症),源文件目录下还放scon脚本做甚,更有甚者,还放了个头文件json_tool.h。作者对项目文件的组织管理能力我真不敢恭维啊!要是我,源文件目录绝对只放源文件,还参杂其它的文件做咩。

源文件目录/src/lib_json/ 不需要要的文件见下图的红框,请把它删掉吧,碍眼!

其中CMakeLists.txt是CMake用于构建项目的脚本,sconscript是scons构建项目的脚本,version.h.in是github自动生成的版本信息。删删删,统统删了,用不到。当然你如果使用cmake来构建项目的话, CMakeLists.txt还是有用的。Linux下,我是不用cmake,虽然它简单好用,但是在每个目录下都要加个CMakeLists.txt,真的很碍眼(个人感觉)。而且CMake最终也是要生成makefile来构建项目,为何不手写一个makefile全部搞定。

第二步: 编译。将上面说明的我们需要的源文件和头文件包含到自己的项目中就可以使用啦。当然你也可以单独将jsoncpp编译成静态链接库或者动态链接库来使用。这里贴一下我用于编译的makefile,喜欢手写makefile的朋友可以通通交流。

##################################
# @brief:make scripts
# @date:2016.06.25
# @author:dablelv
##################################

#environment var
VPATH+=src:src/lib_json

CC:=g++
FLAGS=-g -Wall -std=c++11
INC+=-Iinc
LIBDIR+=

CPPDIRS=src src/lib_json

TARGET:=jsoncpptest.out

CPPS=$(shell for dir in ${CPPDIRS};do echo $${dir}/*.cpp;done)

OBJDIR=obj
OBJS=$(patsubst %.cpp,${OBJDIR}/%.o,$(notdir ${CPPS}))

${TARGET}:${OBJS}
    ${CC} ${FLAGS} ${OBJS} -o $@ ${LIBDIR}

${OBJDIR}/%.o:%.cpp
    ${CC} ${FLAGS} ${INC} -o $@ -c $<

.PHONY:clean
clean:
    rm -f ${TARGET} ${OBJDIR}/*

3.jsoncpp使用实例

下面直接上代码。

#include <string.h>

#include <string>
#include <iostream>
using namespace std;

#include "json/json.h"

struct Student{
    char ID[20];
    char name[10];
    int age;
    int gender;
    char major[10]; 
};

string serializeToJson(const Student& student);
Student deserializeToObj(const string& strJson);

int main(int argc, char** argv) {

    Student student;
    strcpy(student.ID,"312822199204085698");
    strcpy(student.name,"dablelv");
    student.age=18;
    student.gender=0;
    strcpy(student.major,"math");

    string strJson=serializeToJson(student);
    cout<<"strJson:"<<strJson<<endl;

    string strJsonNew="{\"ID\":\"201421031059\",\"name\":\"lvlv\",\"age\":18,\"gender\":0}";
    Student resStudent=deserializeToObj(strJsonNew);
    cout<<"resStudent:"<<endl;
    cout<<"ID:"<<resStudent.ID<<endl;
    cout<<"name:"<<resStudent.name<<endl;
    cout<<"age:"<<resStudent.age<<endl;
    cout<<"gender:"<<resStudent.gender<<endl;
    cout<<"major:"<<resStudent.major<<endl;

    return 0;
}

//@brief:将给定的学生对象序列化为json字符串
//@param:student:学生对象
//@ret:json字符串
string serializeToJson(const Student& student){
    //Json::Value root;
    Json::FastWriter writer;
    Json::Value person;

    person["ID"] = student.ID;
    person["name"] = student.name;
    person["age"]=student.age;
    person["gender"]=student.gender;
    person["major"]=student.major;
    //root.append(person);

    string strJson=writer.write(root);
    return strJson;
}

//@brief:将给定的json字符串反序列化为学生对象
//@param:strJson:json字符串
//@ret:学生对象
Student deserializeToObj(const string& strJson){
    Json::Reader reader;
    Json::Value value;
    Student student;
    memset(&student,0,sizeof(Student));

    if (reader.parse(strJson, value)){
        strcpy(student.ID,value["ID"].asString().c_str());
        strcpy(student.name,value["name"].asString().c_str()); 
        student.age=value["age"].asInt(); 
        student.gender=value["gender"].asInt();
        strcpy(student.major,value["major"].asString().c_str());
    }
    return student;
}

程序输出结果:

上面的major输出之所以为空,是因为json字符串中没有major字段。


参考文献

[1]jsoncpp官网 [2]jsoncpp github [3]native-json benchmark [4]C++ Jsoncpp源代码编译与解析Json

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 将字符转换成带有圆圈的字符

    Dabelv
  • 将字符转换成带有圆圈的字符

    Dabelv
  • Golang json解析与生成

    JSON(Javascript Object Notation)是一种轻量级的数据交换语言,以文字为基础,具有自我描述性且易于让人阅读。尽管JSON是JavaS...

    Dabelv
  • vue3.0 加载json的“另类”方法(非ajax) 定义组件.vue文件

    最近学习vue3.0,在实现一个功能的时候发现一个问题—— 写代码的时候,需要的json太长、太多,和代码放在一起太混乱。看代码总有翻来翻去,又没有“折叠”功...

    用户1174620
  • SpringMVC底层数据传输校验重传方案

    团队的项目正常运行了很久,但近期偶尔会出现BUG。目前观察到的有两种场景:一是大批量提交业务请求,二是生成批量导出文件。出错后,再执行一次就又正常了。

    程序猿讲故事
  • SQL语句汇总(二)——数据修改、数据查询

    注:修改多个列的值时用逗号隔开。要想设置某一列的值为空,只需让<列名>=NULL 即可。WHERE表示过滤条件。

    _DIY
  • Elasticsearch使用REST API实现全文检索

    通过rest api添加检索数据,阅读官方文档可以发现,elasticsearch支持动态映射,但是其中有不少问题,且听慢慢详解。 本文主要讲述三点内容: ...

    用户1154259
  • Flutter Json渐进式解析(下)

    前面几种Json数据格式最外层都是JsonObject,而下面这种,最外层就是一个JsonArray,数据如下所示。

    用户1907613
  • 归并排序的递归实现

    可以看到,每次程序都将数字两两分组(n/2个组),组内单独排序,最后将这些组两两合并,生成n/4个组,组内再单独排序。直到最后只剩下一个组为止。

    lollipop72
  • 修改Github 项目中显示设定的语言

    比如你有一个php项目,如果里面有过多的js,css,html代码,github就会给你的项目设定的语言为js,css,html, 而不是你的项目语言

    Alone88

扫码关注云+社区

领取腾讯云代金券