首页
学习
活动
专区
圈层
工具
发布
社区首页 >专栏 >Linux系统C++开发工具(四)—— jsoncpp 使用指南

Linux系统C++开发工具(四)—— jsoncpp 使用指南

作者头像
用户11719974
发布2025-11-15 10:46:12
发布2025-11-15 10:46:12
50
举报

概念简述

JSON 的全称是 JavaScript Object Notation。

  • 核心定义:它是一种轻量级的数据交换格式,用来做序列化与反序列化

  Json格式可辅助解决 TCP 传输中的粘包问题,粘包——发送方连续发送多个独立数据包,接收方可能一次性读取到合并数据。比如发送6和7,接收到67。   对于粘包问题,可以在数据之间添加一些标识符来区分它们,比如": ",当接收到完整的数据包后根据这些标识符就可以区分出它们。这个操作就是序列化和反序列化。   数据包以什么格式传递是通信双方(服务器和客户端)约定好的,即协议。序列化和反序列化就是协议的一部分。 例如Json会以这样的格式记录一个学生信息:

代码语言:javascript
复制
{
		"姓名": "张三",
		"年龄": 18,
		"成绩": [88.8,89,69.5]
}

一个花括号{}就是一个对象,方括号[ ]就是数组,对象之间可以嵌套,如一个班级的信息:

代码语言:javascript
复制
[{},{},{}]

Json的安装:

代码语言:javascript
复制
ubuntu: sudo apt-get install libjsoncpp-dev
Centos: sudo yum install jsoncpp-devel

数据存取

Json::Value类   这个类用于进行数据中间存储。将多个字段数据进行序列化,需要将数据存储到Value对象中,若要将一个json格式的字符串反序列化,需要用该类对象来接收结果。 数据的存储   Json::Value类里面重载了=运算符和[]运算符,可以如下方式进行数据存储:

代码语言:javascript
复制
Json::Value val;
val["姓名"]="张三";
val["年龄"]=18;
val["成绩"].append(88); //如果要存储一个数组需要用append接口
val["成绩"].append(95.5); //再次调用来新增第二个元素
......

数据的访问   数据访问时同样用方括号[ ]运算符进行访问。在访问时用的类型有多中多样,通过接口调用来转化,如下:

代码语言:javascript
复制
val["姓名"].asString();
val["年龄"].asInt();
val["成绩"][0].asFloat();
val["成绩"][1].asFloat();
......
//对于数组的数据也可提供size()获取元素个数,然后进行遍历。

序列化

StreamWriter类 该类提供了write()接口,用来做序列化

int write(Value const& root, std::ostream* sout)

  • root:一个Json::Value对象,即需要序列化的数据
  • sour:该参数用来接收序列化的结果,通常传入std::stringstream类型
  • 注意:StreamWriter类不能自己构造,需要使用StreamWriterBulider工厂类对象构造。通过工厂类对象里面的newStreamWriter接口生成StreamWriter对象。

反序列化

CharReader类 该类提供parse()接口,用来做反序列化

bool parse(char const* beginDoc, char const* endDoc, Value* root, std::string* errs)

  • beginDoc:JSON 字符串的起始指针
  • endDoc:JSON 字符串的结束指针
  • root:输出型参数,解析后的 JSON 对象
  • errs:输出型参数,错误信息

CharReader类也不能自己构造,需要使用CharReaderBuilder工厂类对象构造,通过工厂类里面的newCharReader接口生成CharReader类。

示例

main.cc文件

代码语言:javascript
复制
#include <json/json.h>
#include <iostream>
#include <sstream>
bool Serialize(const Json::Value& val,std::string& out)
{
    //通过工厂类构建StreamWriter
    auto nsw = Json::StreamWriterBuilder().newStreamWriter();//因为工厂类在该场景中不在使用,所以构造临时对象。
    std::stringstream ss;
    int ret = nsw->write(val,&ss);
    if(ret != 0)
    {
        std::cout<<"序列化失败";
        return false;
    }
    out = ss.str();
    return true;
}
bool UnSerialize(const std::string& str,Json::Value& val)
{
    auto crb = Json::CharReaderBuilder().newCharReader();
    std::string erro;
    bool ret = crb->parse(str.c_str(),str.c_str()+str.size(),&val,&erro);
    if(ret == false)
    {
        std::cout<<"反序列化失败"<<std::endl;
        return false;
    }
    return true;
}
int main()
{
    //构建数据
    Json::Value val;
    val["姓名"] = "张三";
    val["年龄"] = 18;
    val["成绩"].append(90);
    val["成绩"].append(90.5);
    val["成绩"].append(89.5);
    
    //序列化
    std::string out;
    Serialize(val,out);
    std::cout<<out<<std::endl;
    
    //反序列化
    Json::Value root;
    UnSerialize(out,root);
    std::cout<<root["姓名"].asString()<<std::endl;
    std::cout<<root["年龄"].asInt()<<std::endl;
    for(int i=0;i<root["成绩"].size();i++)
    {
        std::cout<<root["成绩"][i]<<" ";
    }
    std::cout<<std::endl;
    return 0;
}

makefile文件

代码语言:javascript
复制
test:main.cc
	g++ $^ -o $@ -std=c++17 -ljsoncpp

执行结果:

在这里插入图片描述
在这里插入图片描述

提示:显示的乱码是数据在转化为Json串后对中文进行了转义存储。

非常感谢您能耐心读完这篇文章。倘若您从中有所收获,还望多多支持呀!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-11-14,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 概念简述
  • 数据存取
  • 序列化
  • 反序列化
  • 示例
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档