当前位置: 首页 > news >正文

【网络】自定义协议 | 序列化和反序列化 | Jsoncpp

本文首发于 慕雪的寒舍

以tcpServer的计算器服务为例,实现用jsoncpp来进行序列化和反序列化

阅读本文之前,请先阅读 自定义协议 | 序列化和反序列化 | 以tcpServer为例

1.安装jsoncpp

我所用的系统是centos7.6,先用下面的命令查找相关的包

sudo yum list | grep jsoncpp-devel

显示出来的包如下

Repository epel is listed more than once in the configuration
jsoncpp-devel.x86_64                0.10.5-2.el7           @epel 

随后安装这个包

sudo yum install jsoncpp

不知道为什么,安装jsoncpp-devel.x86_64显示找不到相关包

或者采用下面的两个命令

sudo yum install epel-release
sudo yum install jsoncpp-devel

如果执行完毕后显示找不到jsoncpp,尝试重新安装epel-release

image-20230404123816637

刚开始我显示epel已经安装,但是找不到jsoncpp这个包,我在重新安装了epel后就能正常安装了

sudo yum rm epel-release
sudo yum install epel-release
sudo yum install jsoncpp-devel

如下图,正常查找到并安装完毕

image-20230404123834144

在centos8下,安装完毕后路径如下

$ ls /usr/include/json
allocator.h  assertions.h  autolink.h  config.h  features.h  forwards.h  json.h  reader.h  value.h  version.h  writer.h

1.1 什么是json?

json是一个kv键值对的序列化方式,每一个key都对应了一个value

{"data": "value"
}

这就有点类似c++中的map,不过json能做的更多

{"data1": "value","data2": {"key1":"value1","key2":"value2","key3":0,"key4":true}
}

json可以在里面嵌套添加更多的内容,不管是字符串还是整形,还能是bool类型的true/false

因为json对反序列化序列化的控制很是不错,可读性也很好,所以被广泛使用!

相比于自己写一个序列化方式,直接用别人的轮子,也不错👻

类似的序列化框架还有xml

2.代码示例

2.1 序列化

相比我们自己写的序列化方式,json的使用简单多了

void serialize(std::string& out)
{//使用jsoncpp的代码Json::Value root;root["x"] = _x;root["y"] = _y;root["op"] = _ops;Json::FastWriter fw; // 这个是写成一行,对于计算机来说处理的负担小// Json::StyledWriter fw; // 这个会进行格式化,更好看(但是内容没差距)out = fw.write(root);
}

对于fw.write(root),其返回值是一个string,也方便我们接收

image-20230212135543386

其中FastWriterStyledWriter有一点区别,如下

//FastWriter
{"data": "value"}
//StyledWriter
{"data": "value"
}

简单来说,StyledWriter会对我们的kv键值对进行格式化,更方便人类的阅读。而FastWriter是直接写成一整行,在传输的时候会方便一点(因为不需要\n

2.2 反序列化

反序列化需要一个Reader来读取字符串,并将其内容根据键值隐射给成员变量

bool deserialize(const std::string &in)
{//jsonJson::Value root;Json::Reader rd;rd.parse(in, root);_x = root["x"].asInt();_y = root["y"].asInt();_ops = root["op"].asInt();
}

3.测试

因为jsoncpp是一个第三方库,我们链接的时候需要加上命令-ljsoncpp

tcpServer:tcpServer.cppg++ -o $@ $^ -std=c++11 -lpthread -ljsoncpp

发送消息后,可以看到,json帮我们格式化为如下形式的字符串

{"op":43,"x":1,"y":200}
{"op":43,"x":333,"y":234}

image-20230212140924067

4.gcc给予宏定义

为了方便对序列化采用的方式进行控制,这里我使用了预处理指令ifdef/endif来进行判断,只要我们在文件头定义了MYPROTOCOL,这里就会采用我们自己的写的序列化方式,否则采用json

#define MYPROTOCOL 1  //如果define了这个,那就使用自己的代码

image-20230212182113854

4.1 命令行

但是在文件里面修改define还是不太方便,我们可以直接采用gcc的命令行参数的方式,进行define的插入(这么做之前,要先删除文件中对MYPROTOCOL的define)

g++ -DMYPROTOCOL tcpServer.cpp -o tcpServer -lpthread -ljsoncpp
g++ -DMYPROTOCOL tcpClient.cpp -o tcpClient -lpthread -ljsoncpp

可以看到,不加编译指令编译出来的服务器,采用的是json的方式来序列化

image-20230212182627766

添加了之后,就是用我们自己写的序列化方式来序列化了

image-20230212182919150

4.2 makefile

因此,我们可以修改makefile来实现这一点

.PHONY:all
all:tcpClient tcpServer
MYSELF=-DMYPROTOCOLtcpClient: tcpClient.cppg++ $(MYSELF) -o $@ $^ -std=c++11 -lpthread -ljsoncpp
tcpServer:tcpServer.cppg++ $(MYSELF) -o $@ $^ -std=c++11 -lpthread -ljsoncpp.PHONY:clean
clean:rm -f tcpClient tcpServer

当我们需要用自己协议的时候,就在最前面加上

MYSELF=-DMYPROTOCOL

否则直接删除这个定义,或者注释掉后面的内容,就能采用jsoncpp

MYSELF=#-DMYPROTOCOL

这样就方便一些了

image-20230212183456024

http://www.lryc.cn/news/114296.html

相关文章:

  • PHP实践:用openssl打造安全可靠的API签名验证系统
  • 每天一道leetcode:剑指 Offer 50. 第一个只出现一次的字符(适合初学者)
  • 【第五章 flutter学习之flutter进阶组件-下篇】
  • 单元测试和集成测试有什么区别
  • 如何实现基于场景的接口自动化测试用例?
  • SAP 开发编辑界面-关闭助手
  • 【el-image图片查看时 样式穿透表格问题】
  • GPT带我学-设计模式-模板模式
  • Windows下调试UEFI程序:Visual Studio调试
  • Vue中监听路由参数变化的几种方式
  • angular——子组件如何接收父组件的动态传值
  • php 桥接模式
  • Android 13 Hotseat定制化修改——004 hotseat布局位置
  • 海外版金融理财系统源码 国际投资理财系统源码 项目投资理财源码
  • 洛谷P1162 - 填涂颜色
  • 设计模式十一:外观模式(Facade Pattern)
  • GIS和倾斜摄影的关系?
  • 【CI/CD】图解六种分支管理模型
  • LeetCode105. 从前序与中序遍历序列构造二叉树
  • 编码技巧——Sentinel的blockHandler与fallback
  • 最新成果展示:GaN基Micro-LED热学模型数据库的开发及应用
  • 【Vue3】动态组件
  • Java超级玛丽小游戏制作过程讲解 第五天 创建并完成常量类04
  • 设置浏览器兼容
  • Java # List
  • git原理与使用
  • 【C语言题解】将一句话的单词进行倒置,标点不倒置。
  • Postman 的简单使用
  • 在CentOS7安装部署GitLab服务
  • 订单系统就该这么设计,稳的一批~