【计算机网络】应用层自定义协议

news2025/1/12 4:09:32

自定义协议

  • 一、为什么需要自定义协议?
  • 二、网络版计算器
    • 1. 基本要求
    • 2. 序列化和反序列化
    • 3. 代码实现
      • (1)封装 socket
      • (2)定制协议和序列化反序列化
      • (3)客户端
      • (4)计算器服务端
      • (5)TCP服务端
      • (6)启动服务器
    • 4. 使用 JSON 进行序列化和反序列化
      • (1)安装 JSON 库
      • (2)测试 JSON
      • (3)在网络计算器中使用 JSON

一、为什么需要自定义协议?

我们上个知识点编写的TCP中,TCP是面向字节流的,我们怎么保证读取上来的数据是一个完整的报文呢?其实我们写的代码中不能保证这个问题,所以代码是有BUG的。TCP 叫做传输控制协议,也就是什么时候发送给对方,发多少,出错了怎么办,完全是由发送方的 TCP 协议来定!当我们使用 write() 函数向 sockfd 中写入数据时,数据不一定已经发给对方了,它的作用其实就是用户到内核的拷贝!这跟我们以前学的向文件中写入是一样的,我们将数据通过 fd 写入到内核的缓冲区,通过操作系统向磁盘中刷新缓冲区的内容。所以真正决定网路收发的协议是由 TCP 决定的!

而对于接收缓冲区来说,我们使用 read() 读取上来的数据就完全不确定了。所以我们在应用层就需要把协议定好,把协议定好才能更好的进行读上来的数据的分析!

所以回到最开始的问题,我们在进行读取的时候,怎么保证读取上来的数据是一个完整的报文呢?对于发送方,是将数据拷贝到它的 TCP 的发送缓冲区了,我们想怎么发完全不由应用层决定,是由 TCP 决定的,所以对方在它的接收缓冲区读上来的数据有可能是我们发送的一部分!所以对方在读的时候,怎么保证读到的是完整的呢?这就需要协议来进行定制了!

在这里插入图片描述

所以我们可以规定好通信双方只能使用固定大小的报文,即我们自己使用自定义协议。

二、网络版计算器

1. 基本要求

例如,我们需要实现一个服务器版的加法器。我们需要客户端把要计算的两个加数发过去,然后由服务器进行计算,最后再把结果返回给客户端。

在应用层定协议,我们通常需要一个比较关键的字段。首先,协议本身就是一种“约定”,假设我们以实现网络版计算器为例,那么我们需要定义的第一个协议就是 request,代表需要相加的两个数和一个操作符,如下:

				struct request
				{
				    int x;
				    int y;
				    char op;
				};

另外还需要定义另一个协议为 response,代表运算结果和正确性,如下:

				struct response
				{
				    int result;
				    int code;
				};

所以每一个结构体的每一个字段,每一个字段里的每一种值,我们都是要让客户端和服务器双方约定好的,约定好之后,我们使用结构化的方式,把约定表达出来,这就叫做我们定义出来的协议。

2. 序列化和反序列化

当我们向对方发信息时, 不仅仅只包含我们所发的信息,还有对应的头像,昵称和时间等等,实际上这些都是一个个的字符串,所以对方会收到四个字符串,但是肯定不能一个个发,是要把它们看作一个整体发给对方;而对方在收到这个整体的字符串后,就要将这个整体的字符串反向的转化成四个字符串,解析成信息内容、头像、昵称和时间。

那么怎么将这些信息看作一个整体呢?我们可以把需要发送的一个信息看作是一个结构体,其中这个结构体中有四个字段,分别代表上面的四个字符串;然后我们再把这个结构化的数据转化成为一个字符串,紧接着将这个字符串整体通过网络发送给对方主机,当对方主机收到这个字符串后,需要将这个字符串解析成为相同类型的结构化数据!在这个消息转化的过程,也是规定出来客户端和服务器双方约定出来的一种通用型的结构体,这就叫做双方定义出来的聊天协议。而在网络通信的时候,整个结构化的数据,把它多个字符串转化成一个字符串整体,这个过程我们称为序列化!而对方把一个字符串整体打散称为多个字符串这个过程称为反序列化

在这里插入图片描述

而以上的过程我们可以看作两层,一层是协议的定制,另一层是序列化和反序列化,如下图:

在这里插入图片描述

那么为什么需要进行序列和反序列化呢?主要是为了方便网络进行收发!

所以根据我们自定义的协议和序列化反序列化,我们的网络版计算机的简略流程如下:

在这里插入图片描述

下面我们根据上图的流程图简易实现一个网络版的计算器。

3. 代码实现

(1)封装 socket

每次提供网络通信都要重新编写 socket 套接字的代码,所以我们现在这里对 socket 进行一下简单的封装,代码如下:

				#pragma once
				
				#include <iostream>
				#include <string>
				#include <cstring>
				#include <unistd.h>
				#include <sys/types.h>
				#include <sys/stat.h>
				#include <sys/socket.h>
				#include <arpa/inet.h>
				#include <netinet/in.h>
				
				#include "log.hpp"
				
				enum
				{
				    SocketErr = 2,
				    BindErr, 
				    ListenErr,
				};
				
				const int backlog = 10;
				
				class Sock 
				{
				public:
				    Sock()
				    {}
				    ~Sock()
				    {}
				
				public:
				    void Socket()
				    {
				        _sockfd = socket(AF_INET, SOCK_STREAM, 0);
				        if(_sockfd < 0)
				        {
				            lg(Fatal, "socket error, %s: %d", strerror(errno), errno);
				            exit(SocketErr);
				        }
				    }
				
				    void Bind(uint16_t port)
				    {
				        sockaddr_in local;
				        memset(&local, 0, sizeof(local));
				        local.sin_addr.s_addr = INADDR_ANY;
				        local.sin_family = AF_INET;
				        local.sin_port = htons(port);
				
				        if(bind(_sockfd, (const sockaddr*)&local, sizeof(local)) < 0)
				        {
				            lg(Fatal, "bind error, %s: %d", strerror(errno), errno);
				            exit(BindErr);
				        }
				    }
				
				    void Listen()
				    {
				        if(listen(_sockfd, backlog) < 0)
				        {
				            lg(Fatal, "listen error, %s: %d", strerror(errno), errno);
				            exit(ListenErr);
				        }
				    }
				
				    int Accept(std::string* client_ip, uint16_t* client_port)
				    {
				        sockaddr_in peer;
				        socklen_t len = sizeof(peer);
				        int newfd = accept(_sockfd, (sockaddr*)&peer, &len);
				        if(newfd < 0)
				        {
				            lg(Warning, "accept error, %s: %d", strerror(errno), errno);
				            return -1;
				        }
				        char buffer[64];
				        inet_ntop(AF_INET, &peer.sin_addr, buffer, sizeof(buffer));
				        *client_ip = buffer;
				        *client_port = ntohs(peer.sin_port);
				
				        return newfd;
				    }
				
				    void Close()
				    {
				        close(_sockfd);
				    }
				 
				    bool Connect(std::string serverip, uint16_t serverport)
				    {
				        sockaddr_in peer;
				        memset(&peer, 0, sizeof(peer));
				        inet_pton(AF_INET, serverip.c_str(), &(peer.sin_addr));
				        peer.sin_family = AF_INET;
				        peer.sin_port = htons(serverport);
				
				        int n = connect(_sockfd, (const sockaddr*)&peer, sizeof(peer));
				        if(n < 0)
				        {
				            lg(Fatal, "connect error, %s: %d", strerror(errno), errno);
				            return false;
				        }
				
				        return true;
				    }
				
				    int GetFd()
				    {
				        return _sockfd;
				    }
				private:
				    int _sockfd;
				};

(2)定制协议和序列化反序列化

在进行定制协议的时候,为了保证对方接受时是一个完整的报文,也就是当对方进行读取时,对方怎么知道是一个报文多大多长呢?所以我们需要使用分隔符将报文和报文之间分隔开来,比如可以使用 \n,也就是使用 \n 对报文之间进行分隔。 但是我们在实现的时候,在报文前再加上一个字段,就是代表有效报文的长度,长度和报文之间也是使用 \n 进行分隔。那么在进行读取的时候,在遇到第一个 \n 之前,就是该报文的长度,然后根据长度去读取报文,就能保证读取到一个完整的报文,当遇到第二个 \n 就代表本次读取完毕,进行下一次读取。

Request

				const std::string blank_space_sep = " ";
				const std::string protocol_sep = "\n";

				// 定制协议
				class Request
				{
				public:
				    Request(int x, int y, char op)
				        : _x(x), _y(y), _op(op)
				    {}
				
				    Request()
				    {}
				
				public:
				    // 序列化
				    bool Serialize(std::string *out)
				    {
				        // 构建报文的有效载荷
				        // struct => string
				        // "len"\n"x op y"\n
				        std::string s = std::to_string(_x);
				        s += blank_space_sep;
				        s += _op;
				        s += blank_space_sep;
				        s += std::to_string(_y);
				
				        *out = s;
				        return true;
				    }
				
				    // 反序列化
				    bool Deserialize(const std::string &in) // "x op y"
				    {
				        size_t left = in.find(blank_space_sep);
				        if(left == std::string::npos) return false;
				        std::string part_x = in.substr(0, left);
				
				        size_t right = in.rfind(blank_space_sep);
				        if(right == std::string::npos) return false;
				        std::string part_y = in.substr(right + 1);
				
				        if(left + 1 != right - 1) return false;
				        
				        _op = in[left + 1];
				        _x = std::stoi(part_x);
				        _y = std::stoi(part_y);
				        return true;
				    }
				
				    void DebugPrint()
				    {
				        std::cout << "新请求构建完成: " << _x << _op << _y << "=?" << std::endl; 
				    }
				
				public:
				    int _x;
				    int _y;
				    char _op;
				};

Response

				class Response
				{
				public:
				    Response(int result, int code)
				        : _result(result), _code(code)
				    {}
				
				    Response()
				    {}
				
				public:
				    // 序列化
				    bool Serialize(std::string *out)
				    {
				        // "len"\n"result code"\n
				        std::string s = std::to_string(_result);
				        s += blank_space_sep;
				        s += std::to_string(_code);
				
				        *out = s;
				        return true;
				    }
				
				    // 反序列化
				    bool Deserialize(const std::string &in)     // "result code"
				    {
				        size_t pos = in.find(blank_space_sep);
				        if(pos == std::string::npos) return false;
				        std::string part_left = in.substr(0, pos);
				        std::string part_right = in.substr(pos + 1);
				
				        _result = std::stoi(part_left);
				        _code = std::stoi(part_right);
				
				        return true;
				    }
				
				    void DebugPrint()
				    {
				        std::cout << "结果响应完成, result: " << _result << ", code: "<< _code << std::endl; 
				    }
				
				public:
				    int _result;
				    int _code; // 表示结果的准确性
				};

下面对封装报头和提取报文也进行简单封装:

				// 封装报头   "x op y" =>  "len"\n"x op y"\n
				std::string Encode(std::string &content)
				{
				    std::string package = std::to_string(content.size());
				    package += protocol_sep;
				    package += content;
				    package += protocol_sep;
				
				    return package;
				}
				 
				// 提取报文   "len"\n"x op y"\n => "x op y"
				bool Decode(std::string& package, std::string* content)
				{
				    size_t pos = package.find(protocol_sep);
				    if(pos == std::string::npos) return false;
				    std::string len_str = package.substr(0, pos);
				    size_t len = stoi(len_str);
				
				    // package = len_str + content_str + 2(\n)
				    // size_t total_len = len_str.size() + len + 2*protocol_sep.size();
				    size_t total_len = len_str.size() + len + 2;
				    if(package.size() < total_len) return false;
				
				    *content = package.substr(pos + 1, len);
				
				    // 如果已经得到一个完整的报文,需要移除这个报文
				    package.erase(0, total_len);
				    return true;
				}

(3)客户端

客户端首先创建需求,然后将需求序列化,并添加报头后通过网络进行发送,当服务端把计算结果返回响应时,客户端进行读取,将数据提取报文,并反序列化得到结果。

				#include <iostream>
				#include <string>
				#include <ctime>
				#include <unistd.h>
				#include "Socket.hpp"
				#include "Protocol.hpp"
				
				using namespace std;
				
				void Usage(const string& str)
				{
				    cout << "\nUsage: " << str << " serverip serverport\n\n" << endl;
				}
				
				int main(int argc, char* argv[])
				{
				    if(argc != 3)
				    {
				        Usage(argv[0]);
				        exit(0);
				    }
				
				    uint16_t server_port = stoi(argv[2]);
				    string server_ip = argv[1];
				
				    Sock _sock;
				    _sock.Socket();
				    bool ret = _sock.Connect(server_ip, server_port); 
				    if(!ret)
				    {
				        cerr << "client connect error" << endl;
				        return 1;
				    }
				
				    srand(time(nullptr));
				    int cnt = 1;
				    string operas = "+-*/%=^&";
				
				    string inbuffer_stream;
				    while(cnt <= 10)
				    {
				        cout << "====================第" << cnt << "次测试......" << endl;
				        int x = rand() % 100 + 1;
				        usleep(1000);
				        int y = rand() % 100;
				        usleep(1000);
				        char op = operas[rand() % operas.size()];
				        Request req(x, y, op);
				        req.DebugPrint();
				
				        string package;
				        req.Serialize(&package);
				        package = Encode(package);
				
				        std::cout << "最新请求: \n" << package;
				        write(_sock.GetFd(), package.c_str(), package.size());
				
				        char buffer[128];
				        size_t n = read(_sock.GetFd(), buffer, sizeof(buffer));
				
				        if(n > 0)
				        {
				            buffer[n] = 0;
				            inbuffer_stream += buffer;  // "len"\n"result code"\n
				            std::cout << inbuffer_stream << std::endl;
				            string content;
				            bool ret = Decode(inbuffer_stream, &content);   // "result code"
				            if(!ret)
				            {
				                cerr << "Decode err" << endl;
				                return 2;
				            }
				
				            Response resp;
				            ret = resp.Deserialize(content);
				            if(!ret)
				            {
				                cerr << "Deserialize err" << endl;
				                return 3;
				            }
				            resp.DebugPrint();
				        }
				        cnt++;
				        cout << "=================================" << endl;
				        sleep(1);
				    }
				
				    _sock.Close();
				    return 0;
				}

(4)计算器服务端

计算器服务端的 Calculator 方法对 package 进行提取报文,获取到需要计算的数据,然后进行反序列化进行计算后,再根据 Response 进行序列化,最后添加报头后返回。

				#pragma once
				#include <string>
				#include <iostream>
				#include "Protocol.hpp"
				
				enum
				{
				    DIV_ERR = 1,
				    MOD_ERR = 2,
				    OP_ERR = 3
				};
				
				class ServerCal
				{
				public:
				    ServerCal()
				    {}
				
				    Response CalculatorHelper(const Request &req)
				    {
				        Response resp(0, 0);
				        switch (req._op)
				        {
				        case '+':
				            resp._result = req._x + req._y;
				            break;
				        case '-':
				            resp._result = req._x - req._y;
				            break;
				        case '*':
				            resp._result = req._x * req._y;
				            break;
				        case '%':
				        {
				            if (req._y == 0)
				                resp._code = MOD_ERR;
				            else
				                resp._result = req._x % req._y;
				        }
				        break;
				        case '/':
				        {
				            if (req._y == 0)
				                resp._code = DIV_ERR;
				            else
				                resp._result = req._x / req._y;
				        }
				        break;
				        default:
				            resp._code = OP_ERR;
				            break;
				        }
				        return resp;
				    }
				
				    // "len"\n"10 + 20"\n
				    std::string Calculator(std::string &package)
				    {
				        std::string content;
				        bool ret = Decode(package, &content); // content = "10 + 20"
				        if (!ret)
				            return "";
				
				        Request req;
				        ret = req.Deserialize(content); // x = 10, y = 20, op = '+'
				        if (!ret)
				            return "";
				
				        content = "";
				        Response resp = CalculatorHelper(req); // result = 30, code = 0
				
				        resp.Serialize(&content);              // content = "30 0"
				        content = Encode(content);             // content = "len"\n"30 0\n"
				
				        return content;
				    }
				
				    ~ServerCal()
				    {}
				};

(5)TCP服务端

TCP服务端获取到新连接后,根据返回的sockfd就可以进行网络通信,也就是获取到客户端的连接请求,紧接着我们创建子进程为其提供服务,首先进行数据读取,将读取到的数据每次添加到 inbuffer_stream 中,每次获取到数据都进行调用计算器服务端的 Calculator 方法,尝试对获取到的数据进行处理,如果处理成功,会在 Decode 方法中将已经提取的报文移除,所以不影响下次读取。当成功调用 Calculator 方法,就将计算结果发送回去。

				#pragma once
				
				#include <signal.h>
				#include <functional>
				
				#include "Socket.hpp"
				#include "log.hpp"
				
				using func_t = std::function<std::string(std::string &)>;
				
				class TcpServer
				{
				public:
				    TcpServer(uint16_t port, func_t callback)
				        : _port(port), _callback(callback)
				    {}
				
				    ~TcpServer()
				    {}
				
				    bool InitServer()
				    {
				        _listen_sock.Socket();
				        _listen_sock.Bind(_port);
				        _listen_sock.Listen();
				        lg(Info, "init server done");
				        return true;
				    }
				
				    void Start()
				    {
				        signal(SIGCHLD, SIG_IGN);
				        signal(SIGPIPE, SIG_IGN);
				        while (true)
				        {
				            // 获取连接
				            std::string client_ip;
				            uint16_t client_port;
				            int sockfd = _listen_sock.Accept(&client_ip, &client_port);
				            if (sockfd < 0)
				                continue;
				            lg(Info, "accept a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip.c_str(), client_port);
				
				            // 提供服务
				            if (fork() == 0)
				            {
				                _listen_sock.Close();
				                std::string inbuffer_stream;
				
				                // 数据计算
				                while (true)
				                {
				                    char buffer[1280];
				                    ssize_t n = read(sockfd, buffer, sizeof(buffer));
				                    if (n > 0)
				                    {
				                        buffer[n] = 0;
				                        inbuffer_stream += buffer;
				
				                        lg(Debug, "debug: \n%s", inbuffer_stream.c_str());
				
				                        while (true)
				                        {
				                            // 如果解析失败,会返回空串
				                            std::string info = _callback(inbuffer_stream);
				                            if (info.empty())
				                                break;
				
				                            write(sockfd, info.c_str(), info.size());
				                        }
				                    }
				                    else if (n == 0)
				                        break;
				                    else
				                        break;
				                }
				                exit(0);
				            }
				            close(sockfd);
				        }
				    }
				
				private:
				    uint16_t _port;
				    Sock _listen_sock;
				    func_t _callback;
				};

(6)启动服务器

				#include "TcpServer.hpp"
				#include "Protocol.hpp"
				#include "ServerCal.hpp"
				
				using namespace std;
				 
				void Usage(const string& str)
				{
				    cout << "\nUsage: " << str << " port\n\n" << endl;
				}
				
				int main(int argc, char* argv[])
				{
				    if(argc != 2)
				    {
				        Usage(argv[0]);
				        exit(0);
				    }
				
				    uint16_t port = stoi(argv[1]);
				
				    ServerCal cal;
				    TcpServer* tsvp = new TcpServer(port, bind(&ServerCal::Calculator, &cal, std::placeholders::_1));
				    tsvp->InitServer();
				    tsvp->Start();
				    
				    return 0;
				}

4. 使用 JSON 进行序列化和反序列化

JSON 其实是一种帮我们进行序列化和反序列化的工具,上面的序列化和反序列化都是我们自己写的,而现在我们可以直接使用 JSON 帮我们完成序列化和反序列化。

(1)安装 JSON 库

我们在 C++ 中想要使用 JSON,首先需要安装 jsoncpp 第三方库,在我们的云服务器上执行指令 sudo yum install jsoncpp-devel -y 即可。

安装成功后,我们可以通过 ls /usr/include/jsoncpp/json/ 查看到我们需要的头文件,下面我们使用到的是 json.h,但是系统默认的搜索路径是 /usr/include/,所以我们可以在包头文件的时候带上路径,也可以在编译选项中添加。

在这里插入图片描述

我们也可以在 /lib64/libjsoncpp.so 路径下找到 JSON 的第三方库,如下:

在这里插入图片描述

(2)测试 JSON

下面我们简单使用一下 JSON,我们先使用一下序列化的功能:

				int main()
				{
				    Json::Value root;
				    root["x"] = 10;
				    root["y"] = 20;
				    root["op"] = '*';
				
				    Json::FastWriter w;
				    string res = w.write(root);
				
				    cout << res << endl;
				
				    return 0;
				}

如上代码,我们创建一个 Value 万能对象,然后建立 k-v 映射关系,接下来创建一个 FastWriter 的对象,调用对象中的 write() 方法即可进行序列化,结果如下:

在这里插入图片描述

另外,在序列化的时候,我们还可以创建 StyledWriter 的对象,这种是按照特定风格形成的字符串,如下:

在这里插入图片描述

接下来我们进行反序列化,代码如下:

				int main()
				{
				    Json::Value root;
				    root["x"] = 10;
				    root["y"] = 20;
				    root["op"] = '*';
				
				    Json::FastWriter w;
				    // Json::StyledWriter w;
				    string res = w.write(root);
				
				    cout << res << endl;
				
				    Json::Value v;
				    Json::Reader r;
				    r.parse(res, v);
				
				    int x = v["x"].asInt();
				    int y = v["y"].asInt();
				    char op = v["op"].asInt();
				
				    cout << "x = " << x << ", y = " << y << ", op = " << op << endl;
				
				    return 0;
				}

如上代码,在反序列化中我们需要创建一个 Reader 对象,并调用对象中的 parse() 方法,该方法的第一个参数就是需要进行反序列化的字符串,第二个参数就是将反序列化后的字段需要写入到哪个对象中,结果如下:

在这里插入图片描述

(3)在网络计算器中使用 JSON

下面我们对网络版计算器的序列化和反序列化的部分进行修改,我们在该部分添加 JSON 代码,但是我们使用的是条件编译,可以让我们在自己的序列化和反序列化与 JSON 之间进行平滑的切换,代码如下:

				#pragma once
				
				#include <iostream>
				#include <string>
				#include <jsoncpp/json/json.h>
				
				// #define USE_MYSELF 1
				
				const std::string blank_space_sep = " ";
				const std::string protocol_sep = "\n";
				
				// 封装报头   "x op y" =>  "len"\n"x op y"\n
				std::string Encode(std::string &content)
				{
				    std::string package = std::to_string(content.size());
				    package += protocol_sep;
				    package += content;
				    package += protocol_sep;
				
				    return package;
				}
				 
				// 提取报文   "len"\n"x op y"\n => "x op y"
				bool Decode(std::string& package, std::string* content)
				{
				    size_t pos = package.find(protocol_sep);
				    if(pos == std::string::npos) return false;
				    std::string len_str = package.substr(0, pos);
				    size_t len = stoi(len_str);
				
				    // package = len_str + content_str + 2(\n)
				    // size_t total_len = len_str.size() + len + 2*protocol_sep.size();
				    size_t total_len = len_str.size() + len + 2;
				    if(package.size() < total_len) return false;
				
				    *content = package.substr(pos + 1, len);
				
				    // 如果已经得到一个完整的报文,需要移除这个报文
				    package.erase(0, total_len);
				    return true;
				}
				
				// 定制协议
				class Request
				{
				public:
				    Request(int x, int y, char op)
				        : _x(x), _y(y), _op(op)
				    {}
				
				    Request()
				    {}
				
				public:
				    // 序列化
				    bool Serialize(std::string *out)
				    {
				#ifdef USE_MYSELF
				        // 构建报文的有效载荷
				        // struct => string
				        // "len"\n"x op y"\n
				        std::string s = std::to_string(_x);
				        s += blank_space_sep;
				        s += _op;
				        s += blank_space_sep;
				        s += std::to_string(_y);
				
				        *out = s;
				        return true;
				#else
				    Json::Value root;
				    root["x"] = _x;
				    root["y"] = _y;
				    root["op"] = _op;
				    Json::FastWriter w;
				    *out = w.write(root);
				    return true;
				#endif
				    }
				
				    // 反序列化
				    bool Deserialize(const std::string &in) // "x op y"
				    {
				#ifdef USE_MYSELF
				        size_t left = in.find(blank_space_sep);
				        if(left == std::string::npos) return false;
				        std::string part_x = in.substr(0, left);
				
				        size_t right = in.rfind(blank_space_sep);
				        if(right == std::string::npos) return false;
				        std::string part_y = in.substr(right + 1);
				
				        if(left + 1 != right - 1) return false;
				        
				        _op = in[left + 1];
				        _x = std::stoi(part_x);
				        _y = std::stoi(part_y);
				        return true;
				#else
				    Json::Value root;
				    Json::Reader r;
				    r.parse(in, root);
				
				    _x = root["x"].asInt();
				    _y = root["y"].asInt();
				    _op = root["op"].asInt();
				    return true;
				#endif
				    }
				
				    void DebugPrint()
				    {
				        std::cout << "新请求构建完成: " << _x << _op << _y << "=?" << std::endl; 
				    }
				
				public:
				    int _x;
				    int _y;
				    char _op;
				};
				
				class Response
				{
				public:
				    Response(int result, int code)
				        : _result(result), _code(code)
				    {}
				
				    Response()
				    {}
				
				public:
				    // 序列化
				    bool Serialize(std::string *out)
				    {
				#ifdef USE_MYSELF
				        // "len"\n"result code"\n
				        std::string s = std::to_string(_result);
				        s += blank_space_sep;
				        s += std::to_string(_code);
				
				        *out = s;
				        return true;
				#else 
				    Json::Value root;
				    root["result"] = _result;
				    root["code"] = _code;
				    Json::FastWriter w;
				    *out = w.write(root);
				    return true;
				#endif
				    }
				
				    // 反序列化
				    bool Deserialize(const std::string &in)     // "result code"
				    {
				#ifdef USE_MYSELF
				        size_t pos = in.find(blank_space_sep);
				        if(pos == std::string::npos) return false;
				        std::string part_left = in.substr(0, pos);
				        std::string part_right = in.substr(pos + 1);
				
				        _result = std::stoi(part_left);
				        _code = std::stoi(part_right);
				
				        return true;
				#else
				    Json::Value root;
				    Json::Reader r;
				    r.parse(in, root);
				
				    _result = root["result"].asInt();
				    _code = root["code"].asInt();
				    return true;
				#endif
				    }
				
				    void DebugPrint()
				    {
				        std::cout << "结果响应完成, result: " << _result << ", code: "<< _code << std::endl; 
				    }
				
				public:
				    int _result;
				    int _code; // 表示结果的准确性
				};

我们可以通过在编译选项中加上宏定义的选项,使我们更方便地选择哪种序列化和反序列化的方式,例如 makefile 文件中:

				.PHONY:all
				all:servercal clientcal
				
				Flag=-DUSE_MYSELF=1
				Lib=-ljsoncpp
				
				servercal:ServerCal.cc
					g++ -o $@ $^ -std=c++11 $(Lib) $(Flag)
				clientcal:ClientCal.cc 
					g++ -o $@ $^ -std=c++11 $(Lib) $(Flag)
				
				.PHONY:clean 
				clean:
					rm -f servercal clientcal

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1471374.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

通过IP地址确定地理位置

在互联网世界中&#xff0c;每个设备或用户在进行网络通信时&#xff0c;都会使用到一个独特的标识&#xff0c;即IP地址。IP地址不仅用于标识网络中的设备&#xff0c;还可以在一定程度上揭示出这些设备或用户的地理位置信息。本文将详细探讨如何通过IP地址确定地理位置&#…

2-23 switch、JVM内存模型、垃圾回收机制、this、static、变量的分类

文章目录 switch 实现成绩评级JVM内存模型概念栈的特点堆的特点 垃圾回收机制通用的分代垃圾回收机制三种清理算法垃圾回收过程垃圾回收常见的两种检测引用算法内存泄露常见原因 this的用法创建对象的四步 static 静态特点 变量的分类和作用域import switch 实现成绩评级 switc…

Linux基础命令—系统服务

基础知识 centos系统的开机流程 1)通电 2)BIOS硬件检查 3)MBR引导记录 mbr的引导程序 加载引导程序 让硬件加载操作系统内核 MBR在第一个磁盘第一个扇区 总大小512字节 mbr: 1.引导程序: 占用446字节用于引导硬件,加载引导程序 2.分区表: 总共占…

Seata分布式事务实战XATCC模式

目录 XA模式 XA 模式的使用 Spring Cloud Alibaba整合Seata XA TCC模式 TCC模式接口改造 TCC如何控制异常 Spring Cloud Alibaba整合Seata TCC XA模式 整体机制 在 Seata 定义的分布式事务框架内&#xff0c;利用事务资源&#xff08;数据库、消息服务等&#xff09;对…

Mysql学习之数据库事务

事务 数据库事务概述 事务是数据库区别于文件系统的重要特性之一&#xff0c;当有了事务就可以让数据库始终保持一致性。同时还可以通过事务的机制&#xff0c;恢复到某个时间点&#xff0c;这样就可以保证以提交到数据库的修改不会因为系统崩溃而丢失 只有Innodb支持事务的 …

camunda7流程平台技术架构概述

Camunda Platform 是一个基于 Java 的BPMN(流程引擎)、DMN&#xff08;规则引擎&#xff09;、CMMN&#xff08;案例管理&#xff09;的开源框架。主要组件是用 Java 编写的&#xff0c;主要专注于为 Java 开发人员提供在 JVM 上设计、实现和运行业务流程和工作流所需的工具&am…

【TCP/IP】内核网络堆栈

在Linux内核中&#xff0c;网络堆栈&#xff08;network stack&#xff09;是一套实现网络通信功能的软件包&#xff0c;负责处理数据包的发送和接收。网络堆栈按照OSI模型&#xff08;开放式系统互联通信参考模型&#xff09;或TCP/IP模型的层次结构来组织&#xff0c;实现了从…

Linux-部署各类软件(黑马学习笔记)

MYSQL MYSQL5.7版本在CentOS系统安装 注意&#xff1a;安装操作需要root权限 MySQL的安装我们可以通过前面学习的yum命令进行。 安装 1.配置yum仓库 # 更新密钥 rpm --import https://repo.mysql.com/RPM-GPG-KEY-mysql-2022# 安装Mysql yum库 rpm -Uvh http://repo.mysql.…

端口映射的软件有哪些?

端口映射软件是一种实用工具&#xff0c;能够帮助用户在网络中实现远程通信&#xff0c;解决不同地区电脑与电脑、设备与设备、电脑与设备之间的信息传输问题。其中&#xff0c;【天联】组网天联是一款功能强大的端口映射软件&#xff0c;它通过在全国各主要节点部署加速服务器…

介绍 Gradio 与 Hugging Face

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 我们可以使用 Gradio 库为我们的模型构建演示。Gradio 允许您完全使用 Python 为任何机器学习模型构建、自定义和共享基于 Web 的演示。使机器学习模型变得可交互和易于使用。 为什么首先要为您的机器…

贪心算法(算法竞赛、蓝桥杯)--奶牛耍杂技

1、B站视频链接&#xff1a;A24 贪心算法 P1842 [USACO05NOV] 奶牛玩杂技_哔哩哔哩_bilibili 题目链接&#xff1a;[USACO05NOV] 奶牛玩杂技 - 洛谷 #include <bits/stdc.h> using namespace std; const int N50005; struct node{int w,s;bool operator<(node &…

外贸精英催单秘籍:突破观望犹豫,抓住订单黄金时机!

年底将至&#xff0c;对于外贸人来说&#xff0c;这是一个重要的订单冲刺时机。在这个关键时刻&#xff0c;如何向国外客户催单成为一项关键任务。本文将分享催单的技巧&#xff0c;并附带销冠年底工作安排计划。同时&#xff0c;我们将引入Focussend&#xff0c;一款具有邮件自…

Linux软件高级编程-进程基本概念--day6

1.进程&#xff1a; 程序&#xff1a; 存放在外存的一段数据组成的文件 进程&#xff1a; 是一个程序动态执行的过程&#xff0c;包括进程的创建、进程的调度、进程的消亡 2.进程相关命令&#xff1a; 1&#xff09;top&#xff1a; 动态查看当前系统中所有进程信息&#xff08…

Jmeter系列(1)Mac下载安装启动

目录 Jmeter下载安装启动下载启动 Jmeter下载安装启动 注意⚠️&#xff1a;使用jmeter需要有java环境 下载 官网下载地址&#xff1a;https://jmeter.apache.org/ 会看到这里有两个版本&#xff0c;那么有什么区别么&#xff1f; Binaries是可执行版&#xff0c;直接下载解…

Redis哨兵模式和Redis Cluster模式

文章目录 &#x1f50a;博主介绍&#x1f964;本文内容Redis Cluster 模式支持自动故障转移功能吗&#xff1f;Redis Cluster 模式支持自动故障转移功能和哨兵有什么区别&#xff1f;Redis Cluster 模式和哨兵模式&#xff08;Sentinel&#xff09;在自动故障转移方面有一些关键…

Niginx介绍和安装使用

Nginx是什么&#xff1f; Nginx (engine x) 是一个高性能的HTTP和反向代理web服务器&#xff0c;同时也提供了IMAP/POP3/SMTP服务。Nginx是由伊戈尔赛索耶夫为俄罗斯访问量第二的Rambler.ru站点&#xff08;俄文&#xff1a;Рамблер&#xff09;开发的&#xff0c;第一…

什么是Elasticsearch SQL

什么是Elasticsearch SQL 一. 介绍二. SQL 入门 前言 这是我在这个网站整理的笔记,有错误的地方请指出&#xff0c;关注我&#xff0c;接下来还会持续更新。 作者&#xff1a;神的孩子都在歌唱 一. 介绍 Elasticsearch SQL 是一个 X-Pack 组件&#xff0c;允许针对 Elasticsea…

linux系统---防火墙拓展

目录 一、iptables 1.基本语法 2.四表五链——重点记忆 2.1四表 2.2五链 2.3总结 3.iptables选项示例 3.1 -Z 清空流量计数 3.2 -P 修改默认规则 3.3 -D 删除规则 3.4 -R 指定编号替换规则 4.白名单 5.通用匹配 6.示例 6.1添加回环网卡 6.2可以访问端口 6.3 主…

matlab|计及源荷不确定性的综合能源生产单元运行调度与容量配置随机优化模型

目录 1 主要内容 1.1 风光场景聚类 1.2 主模型程序结果 1.3 随机模型和确定性模型对比 1.4 有无储气对比 1.5 煤价灵敏性分析 1.6 甲烷价格灵敏性分析 2 部分程序 3 下载链接 1 主要内容 本程序复现《计及源荷不确定性的综合能源生产单元运行调度与容量配置两阶段随机…

Bert基础(四)--解码器(上)

1 理解解码器 假设我们想把英语句子I am good&#xff08;原句&#xff09;翻译成法语句子Je vais bien&#xff08;目标句&#xff09;。首先&#xff0c;将原句I am good送入编码器&#xff0c;使编码器学习原句&#xff0c;并计算特征值。在前文中&#xff0c;我们学习了编…