【Linux】自定义协议与序列化和反序列化

2024-09-06 11:28

本文主要是介绍【Linux】自定义协议与序列化和反序列化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、自定义协议

1.1 自定义报文格式

       在前面的博客中,我们可以知道在TCP协议中,面向的是字节流;而UDP协议中面向的是数据报。因此,在手写简单的TCP和UDP服务器中,所使用的是接收函数和发送函数不同。因此,在TCP协议中,我们需要分清楚一个完整的报文,并将其分离出来,因此,我们应该如何进行分离出一个完整的报文呢??

       如果一个报文中什么标志也没有,那么必然是不可能将一个完整的报文分离出来。因此,我们可以重新定义一下报文格式:为了简单起见,我们采用LV格式,在后面的学习中,当我们学习了TCP报文和UDP报文后,就会知道报文 = 报头 + 有效载荷。报头中存放有效载荷的长度,我们定义的一个简单的报文格式如图所示:

       在上一篇博客中,我们也简单的学习如何使用Json来进行序列化和反序列化的操作。我们可以在协议层中定义出来将请求和响应的报文进行序列化和反序列化操作,然后将数据通过传输层进行传输。在根据具体的报文的格式进行分离出一个完整的报文。

1.2 协议类(表示层)

       在协议类中,我们创建了请求类和响应类,并且创建出添加报头函数和分解完整报文函数,最后在定义出一个工厂用来创建请求和响应。

1.2.1 请求类

    class Request{public:// 我们自定义协议// 报文 = 包头 + 有效载荷// LV格式  固定字段长——后续字符串的长度  正文内容\n\t// 对报文进行分析// "len\r\n"_x_op_y\r\n"Request(){}Request(int x, int y, char oper): _x(x), _y(y), _oper(oper){}bool Serialize(std::string *out) // 序列化{Json::Value root;root["x"] = _x;root["y"] = _y;root["oper"] = _oper;Json::FastWriter writer;*out = writer.write(root);return true;}bool Deserialize(std::string &in) // 反序列化{Json::Value root;Json::Reader reader;bool ret = reader.parse(in, root);_x = root["x"].asInt();_y = root["y"].asInt();_oper = root["oper"].asInt();return ret;}~Request(){}public:int _x;int _y;char _oper; // "+-*/%"};

1.2.2 响应类

    class Response{public:Response(){}Response(int result, int code): _result(result), _code(code){}bool Serialize(std::string *out) // 序列化{Json::Value root;root["result"] = _result;root["code"] = _code;Json::FastWriter writer;*out = writer.write(root);return true;}bool Deserialize(std::string &in) // 反序列化{Json::Reader reader;Json::Value root;bool ret = reader.parse(in, root);_result = root["result"].asInt();_code = root["code"].asInt();return ret;}~Response(){}public:int _result;int _code;};

1.2.3 添加报文函数

       在将创建出来的请求或者响应进行序列化后,将其进行添加报头方式,将其构造成我们所需要的报文格式,以便在之后进行分离出完整报文的操作。

    const std::string SEP = "\n\t";// 进行拼装报文std::string Encode(const std::string &json_str){int json_str_len = json_str.size();std::string proto_str = std::to_string(json_str_len);proto_str += SEP;proto_str += json_str;proto_str += SEP;return proto_str;}

1.2.4 分解完整报文函数

       解决粘包问题,我们根据报文格式,我们可以得出:在第一次遇见“\n\t”的时候,其前面的字符串会有两种情况:要么是空串,要么是报头。当我们解析到报头后,计算出一个完整的报文的总长度,然后个根据总长度将完整报文解析出来。

    const std::string SEP = "\n\t";// 处理粘包问题std::string Decode(std::string &inbuffer) // const 不能修改{// 先找出SEP,然后截取出来len的长度,最后将完整报文截取出来auto pos = inbuffer.find(SEP);if (pos == std::string::npos)return "";std::string len_str = inbuffer.substr(0, pos);if (len_str.empty())return "";int len = std::stoi(len_str);int total = len + len_str.size() + SEP.size() * 2;if (inbuffer.size() < total)return "";std::string package = inbuffer.substr(pos + SEP.size(), len);inbuffer.erase(0, total); // yichureturn package;}

1.2.5 工厂类

       在这里采用简单工厂模式,我们可以将请求类和响应类设置为私有类,不想外部暴漏;只将工厂类向外暴漏,方便用户进行注册请求和响应。

    // 简单工厂模式class Factory{public:Factory(){srand(time(nullptr) ^ getpid());}// 生产数据   利用随机值将报文填充std::shared_ptr<Request> BuildRequest(){opers = "+-*/%&^";int x = rand() % 10;int y = rand() % 5;char oper = opers[rand() % 8];std::shared_ptr<Request> req = std::make_shared<Request>(x, y, oper);return req;}std::shared_ptr<Response> BuildResponse(){return std::make_shared<Response>();}~Factory() {}private:std::string opers;};

二、复习一下TCP服务器的写法

2.1 回忆一下Socket类

       在上一节课中,我们将Socket进行封装。我们将TCP套接字和UDP套接字的方法进行封装成一个类中,来回忆一下他们的方法总共有哪些?创建套接字,绑定套接字,监听套接字,TCP接收数据,TCP连接,返回套接字,接收数据,发送数据。

// 在抽象类中,我们可以定义出一些虚函数,供TCPSocket和UDPSocket方便构造各自的函数
virtual void CreateSocketOrDie() = 0;             // 创建套接字
virtual void BindSocketOrDie(InetAddr &addr) = 0; // 绑定套接字
virtual void ListenSocketOrDie() = 0;             // 监听套接字
virtual socket_sptr Accepter(InetAddr *addr) = 0; // 接受数据
virtual bool Connector(InetAddr &addr) = 0;       // 连接
virtual int Sockfd() = 0;                         // 返回套接字
virtual int Recv(std::string *out) = 0;           // 接收数据
virtual int Send(const std::string &in) = 0;      // 发送数据

2.2 TcpServer类

       在这个类中,我们可以通过2.1中的TcpSocket类进行创建,我们可以创建出智能指针,通过智能指针来进行管理这个变量,调用里面的函数。在执行函数中,我们可以利用Acceptor函数创建出新的套接字,以便于进行与客户端通信。在这个函数中,我们可以使用多线程来进行。

#pragma once
#include <iostream>
#include <string>
#include <cstring>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <sys/wait.h>
#include <unistd.h>
#include <functional>
#include "Log.hpp"
#include "InetAddr.hpp"
#include "Socket.hpp"
#include <memory>using namespace socket_ns;
using io_service_t = std::function<void(socket_sptr, InetAddr)>;
class TcpServer;
const static int backlog = 16; // 连接队列
class ThreadDate
{
public:ThreadDate(socket_sptr sockfd, InetAddr addr, TcpServer *s) : _sockfd(sockfd), _addr(addr), self(s) {}InetAddr _addr;socket_sptr _sockfd;TcpServer *self;
};class TcpServer
{
public:TcpServer(int port, io_service_t service): _local("0", port), _isrunning(false), _listensock(std::make_unique<TcpSocket>()), _service(service){_listensock->BuildListenSocket(_local); // 创建套接字,绑定套接字,监听套接字}~TcpServer(){}static void *HandlerSock(void *args){pthread_detach(pthread_self());ThreadDate *td = static_cast<ThreadDate *>(args);td->self->_service(td->_sockfd, td->_addr); // 这个地址就是客户端的地址::close(td->_sockfd->Sockfd());delete td;return nullptr;}void Loop(){// 4. 不能直接收数据,先获取连接// accept在通信之前先获取客户端的地址, 返回值是文件描述符// 这个文件描述符是什么?? 每建立一个链接就会有一个套接字, 用于IO操作_isrunning = true;while (_isrunning){// 利用accept函数进行创建出新的套接字InetAddr peeraddr;socket_sptr s = _listensock->Accepter(&peeraddr);if (s == nullptr)continue;// version 2 : 采用多线程pthread_t t;ThreadDate *td = new ThreadDate(s, peeraddr, this);pthread_create(&t, nullptr, HandlerSock, td);}_isrunning = false;}
private:InetAddr _local;std::unique_ptr<Socket> _listensock;bool _isrunning;io_service_t _service;
};

三、业务服务类(应用层)

       为了使业务服务与通信服务进一步解耦合,我们可以将业务服务单独封装成一个类,使得整体布局更加具有层次性。

#pragma once
#include <iostream>
#include "Protocol.hpp"
using namespace protocol_ns;// 解决计算层
class Calculate
{
public:Calculate(){}// 处理函数Response Excute(const Request &req){Response rsp(0, 0);switch (req._oper){case '+':rsp._result = req._x + req._y;break;case '-':rsp._result = req._x - req._y;break;case '*':rsp._result = req._x * req._y;break;case '/':{if (req._y == 0){rsp._code = 1;}else{rsp._result = req._x / req._y;}}break;case '%':{if (req._y == 0){rsp._code = 2;}else{rsp._result = req._x % req._y;}}break;default:rsp._code = 3;break;}return rsp;}~Calculate(){}
private:
};

四、创建客户端(会话层)

       由于之前的套接字封装,我们可以简单地通过智能指针创建出TcpSocket,通过智能指针可以直接创建出客户端套接字并连接服务器成功。我们之后就可以进行通信。

客户端与服务器通信的步骤:创建一个请求;将请求进行序列化;将请求添加报头;发送报文;读取应答;判断应答是否是一个完整的报文;将报文反序列化,最后拿到了结构化的应答。

void Usage(std::string proc)
{std::cout << "Usage:\n\t" << proc << " serverip serverport\n"<< std::endl;
}int main(int argc, char *argv[])
{if (argc != 3){Usage(argv[0]);exit(2);}std::string serverip = argv[1];uint16_t serverport = std::stoi(argv[2]);InetAddr serveraddr(serverip, serverport);std::unique_ptr<Socket> cli = std::make_unique<TcpSocket>();bool res = cli->BuildClientSocket(serveraddr);std::string buffer;while (res){Factory factory;// 1. 创建一个请求auto req = factory.BuildRequest();// 2. 将请求进行序列化std::string request;req->Serialize(&request);// 3. 添加报文长度request = Encode(request);// 4. 发送报文cli->Send(request);// 5. 读取应答int n = cli->Recv(&buffer);if (n < 0){break; // 出错了}buffer = Encode(buffer);// 6. 判断应答是否是一个完整的报文auto resp = factory.BuildResponse();resp->Deserialize(buffer);// 7. 拿到了结构化的应答std::cout << resp->_result << "[" << resp->_code << "]" << std::endl;}return 0;
}

五、总结

       将OSI七层模式与TCP/IP分层模式进行对比,我们会发现TCP/IP分层模型将会话层、表示层与应用层合并为一层。因为应用层必须要在用户层完成,用户决定了与谁构建连接,用户决定了采用什么样的报文格式以及协议,用户决定了采用什么服务。

这篇关于【Linux】自定义协议与序列化和反序列化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1141875

相关文章

Linux使用fdisk进行磁盘的相关操作

《Linux使用fdisk进行磁盘的相关操作》fdisk命令是Linux中用于管理磁盘分区的强大文本实用程序,这篇文章主要为大家详细介绍了如何使用fdisk进行磁盘的相关操作,需要的可以了解下... 目录简介基本语法示例用法列出所有分区查看指定磁盘的区分管理指定的磁盘进入交互式模式创建一个新的分区删除一个存

Linux使用dd命令来复制和转换数据的操作方法

《Linux使用dd命令来复制和转换数据的操作方法》Linux中的dd命令是一个功能强大的数据复制和转换实用程序,它以较低级别运行,通常用于创建可启动的USB驱动器、克隆磁盘和生成随机数据等任务,本文... 目录简介功能和能力语法常用选项示例用法基础用法创建可启动www.chinasem.cn的 USB 驱动

高效管理你的Linux系统: Debian操作系统常用命令指南

《高效管理你的Linux系统:Debian操作系统常用命令指南》在Debian操作系统中,了解和掌握常用命令对于提高工作效率和系统管理至关重要,本文将详细介绍Debian的常用命令,帮助读者更好地使... Debian是一个流行的linux发行版,它以其稳定性、强大的软件包管理和丰富的社区资源而闻名。在使用

Linux Mint Xia 22.1重磅发布: 重要更新一览

《LinuxMintXia22.1重磅发布:重要更新一览》Beta版LinuxMint“Xia”22.1发布,新版本基于Ubuntu24.04,内核版本为Linux6.8,这... linux Mint 22.1「Xia」正式发布啦!这次更新带来了诸多优化和改进,进一步巩固了 Mint 在 Linux 桌面

LinuxMint怎么安装? Linux Mint22下载安装图文教程

《LinuxMint怎么安装?LinuxMint22下载安装图文教程》LinuxMint22发布以后,有很多新功能,很多朋友想要下载并安装,该怎么操作呢?下面我们就来看看详细安装指南... linux Mint 是一款基于 Ubuntu 的流行发行版,凭借其现代、精致、易于使用的特性,深受小伙伴们所喜爱。对

什么是 Linux Mint? 适合初学者体验的桌面操作系统

《什么是LinuxMint?适合初学者体验的桌面操作系统》今天带你全面了解LinuxMint,包括它的历史、功能、版本以及独特亮点,话不多说,马上开始吧... linux Mint 是一款基于 Ubuntu 和 Debian 的知名发行版,它的用户体验非常友好,深受广大 Linux 爱好者和日常用户的青睐,

Linux(Centos7)安装Mysql/Redis/MinIO方式

《Linux(Centos7)安装Mysql/Redis/MinIO方式》文章总结:介绍了如何安装MySQL和Redis,以及如何配置它们为开机自启,还详细讲解了如何安装MinIO,包括配置Syste... 目录安装mysql安装Redis安装MinIO总结安装Mysql安装Redis搜索Red

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

Linux磁盘分区、格式化和挂载方式

《Linux磁盘分区、格式化和挂载方式》本文详细介绍了Linux系统中磁盘分区、格式化和挂载的基本操作步骤和命令,包括MBR和GPT分区表的区别、fdisk和gdisk命令的使用、常见的文件系统格式以... 目录一、磁盘分区表分类二、fdisk命令创建分区1、交互式的命令2、分区主分区3、创建扩展分区,然后

Linux中chmod权限设置方式

《Linux中chmod权限设置方式》本文介绍了Linux系统中文件和目录权限的设置方法,包括chmod、chown和chgrp命令的使用,以及权限模式和符号模式的详细说明,通过这些命令,用户可以灵活... 目录设置基本权限命令:chmod1、权限介绍2、chmod命令常见用法和示例3、文件权限详解4、ch