文章目录
概念简述
JSON 的全称是 JavaScript Object Notation。
- 核心定义:它是一种轻量级的数据交换格式,用来做序列化与反序列化
Json格式可辅助解决 TCP 传输中的粘包问题,粘包------发送方连续发送多个独立数据包,接收方可能一次性读取到合并数据。比如发送6和7,接收到67。
对于粘包问题,可以在数据之间添加一些标识符来区分它们,比如": ",当接收到完整的数据包后根据这些标识符就可以区分出它们。这个操作就是序列化和反序列化。
数据包以什么格式传递是通信双方(服务器和客户端)约定好的,即协议。序列化和反序列化就是协议的一部分。
例如Json会以这样的格式记录一个学生信息:
{
"姓名": "张三",
"年龄": 18,
"成绩": [88.8,89,69.5]
}
一个花括号{}就是一个对象,方括号[ ]就是数组,对象之间可以嵌套,如一个班级的信息:
[{},{},{}]
Json的安装:
bash
ubuntu: sudo apt-get install libjsoncpp-dev
Centos: sudo yum install jsoncpp-devel
数据存取
Json::Value类
这个类用于进行数据中间存储。将多个字段数据进行序列化,需要将数据存储到Value对象中,若要将一个json格式的字符串反序列化,需要用该类对象来接收结果。
数据的存储
Json::Value类里面重载了=运算符和[]运算符,可以如下方式进行数据存储:
json
Json::Value val;
val["姓名"]="张三";
val["年龄"]=18;
val["成绩"].append(88); //如果要存储一个数组需要用append接口
val["成绩"].append(95.5); //再次调用来新增第二个元素
......
数据的访问
数据访问时同样用方括号[ ]运算符进行访问。在访问时用的类型有多中多样,通过接口调用来转化,如下:
json
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文件
cpp
#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文件
makefile
test:main.cc
g++ $^ -o $@ -std=c++17 -ljsoncpp
执行结果:

提示:显示的乱码是数据在转化为Json串后对中文进行了转义存储。
非常感谢您能耐心读完这篇文章。倘若您从中有所收获,还望多多支持呀!
