设计自己的网络通信协议

server/2024/11/25 14:02:33/

文章目录

  • 一、为什么需要设计网络通信协议
      • 1. **标准化通信规则**
      • 2. **确保数据传输的可靠性**
      • 3. **支持网络的多样性和可扩展性**
      • 4. **分层设计,简化复杂性**
      • 5. **实现设备的互操作性**
      • 6. **支持多任务和多应用并发**
      • 7. **提供安全性**
      • 8. **支持不同的通信模式**
      • 总结
  • 二、通信协议设计实战并且适配到TCP和UDP中
      • 通信协议设计
        • 协议字段描述
        • 数据结构定义
      • 功能实现
        • 1. **序列化函数**
        • 2. **反序列化函数**
        • 3. **CRC 校验计算函数**
        • 4. **TCP/UDP 适配**
      • 总结


一、为什么需要设计网络通信协议

设计网络通信协议是为了确保不同的设备、系统或应用能够通过网络进行可靠、规范化、无歧义的数据交换。网络通信协议的设计解决了数据传输中的复杂性和多样性,以下是具体的原因和作用:


1. 标准化通信规则

  • 问题:不同设备和系统可能使用不同的硬件、软件和操作方式。如果没有统一的规则,设备之间将无法理解对方发送的数据。
  • 解决方案:协议规定了如何发送数据(数据格式、编码方式)、如何接收数据(确认、重传机制)、如何处理错误等,确保所有参与通信的实体可以无缝协作。

2. 确保数据传输的可靠性

  • 问题:网络传输过程中可能出现数据丢失、重复、延迟或损坏。
  • 解决方案通信协议通过错误检测(如校验和)、数据分片与重组、重传机制等方法,提高数据传输的可靠性,确保数据从发送方到接收方的完整性和一致性。

3. 支持网络的多样性和可扩展性

  • 问题:网络环境复杂多样,包括局域网(LAN)、广域网(WAN)、无线网络等,不同网络类型可能有不同的特性。
  • 解决方案:协议(如TCP/IP)抽象出通用规则,隐藏底层网络的差异性,提供统一的通信标准,使不同网络类型的设备能够互通。

4. 分层设计,简化复杂性

  • 问题:网络通信涉及多个层次的问题(如硬件连接、数据传输、应用通信),直接设计全功能的网络通信系统复杂且低效。
  • 解决方案:采用分层模型(如OSI模型或TCP/IP模型),将通信分为物理层、链路层、网络层、传输层、应用层等,每一层关注特定功能,便于开发、维护和扩展。

5. 实现设备的互操作性

  • 问题:现代网络由不同厂商的设备组成,如路由器、交换机、服务器和终端设备。如果没有统一的通信规则,它们无法协同工作。
  • 解决方案:协议定义了设备之间的通信规则,确保不同厂商的设备可以互操作,例如HTTP用于浏览器与服务器通信,SMTP用于电子邮件传输。

6. 支持多任务和多应用并发

  • 问题:网络上的设备需要同时处理多个任务,例如浏览网页、视频通话、文件下载等。
  • 解决方案:协议通过端口号、会话管理和流量控制机制,支持多任务和多应用的高效并发通信。

7. 提供安全性

  • 问题:网络通信容易受到攻击,如数据窃取、伪装、篡改等。
  • 解决方案:许多通信协议(如HTTPS、SSL/TLS)内置了加密、身份验证和完整性校验等机制,保护数据传输的安全性。

8. 支持不同的通信模式

  • 问题:网络通信可能是单向、双向、广播、组播等多种模式。
  • 解决方案:协议设计中支持多种通信模式,例如:
    • TCP:面向连接、可靠的双向通信。
    • UDP:无连接、快速的单向或多点通信。

总结

设计网络通信协议是为了建立一个标准化、可靠、安全的通信基础,使不同设备、系统和应用能够高效地协作和互通。通信协议是现代网络运行的核心,没有协议,全球互联网和数字通信将无法实现。

二、通信协议设计实战并且适配到TCP和UDP中

以下是一个使用 C语言 设计的通信协议,包含消息头部数据、数据类型、数据长度、数据主体以及 CRC 校验值。协议使用结构体封装,并设计了适用于 TCPUDP 的传输方式。


通信协议设计

协议字段描述
字段名称类型描述
Headeruint32_t消息头(标识协议类型或版本,固定值)。
Typeuint16_t数据类型(如文本、文件、命令等)。
Lengthuint32_t数据主体的长度(单位:字节)。
CRCuint32_tCRC 校验值,用于校验数据完整性。
Payloadchar[]数据主体(可变长度)。

数据结构定义
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h> // 用于字节序转换// 定义协议结构体
typedef struct {uint32_t header;    // 消息头uint16_t type;      // 数据类型uint32_t length;    // 数据长度uint32_t crc;       // CRC 校验值char payload[];     // 数据主体(可变长度)
} Packet;

功能实现

1. 序列化函数

将结构体转换为字节流以便通过网络传输。

// 序列化函数:将 Packet 结构体转为字节流
void* serializePacket(const Packet* packet, size_t* bufferSize) {*bufferSize = sizeof(Packet) + packet->length;void* buffer = malloc(*bufferSize);if (!buffer) {perror("Failed to allocate memory");return NULL;}char* ptr = buffer;// 写入头部数据,转为网络字节序uint32_t header_be = htonl(packet->header);memcpy(ptr, &header_be, sizeof(uint32_t));ptr += sizeof(uint32_t);// 写入数据类型,转为网络字节序uint16_t type_be = htons(packet->type);memcpy(ptr, &type_be, sizeof(uint16_t));ptr += sizeof(uint16_t);// 写入数据长度,转为网络字节序uint32_t length_be = htonl(packet->length);memcpy(ptr, &length_be, sizeof(uint32_t));ptr += sizeof(uint32_t);// 写入 CRC 校验值,转为网络字节序uint32_t crc_be = htonl(packet->crc);memcpy(ptr, &crc_be, sizeof(uint32_t));ptr += sizeof(uint32_t);// 写入数据主体memcpy(ptr, packet->payload, packet->length);return buffer;
}

2. 反序列化函数

将字节流还原为结构体。

// 反序列化函数:将字节流转回 Packet 结构体
Packet* deserializePacket(const void* buffer) {const char* ptr = buffer;// 读取头部数据Packet* packet = malloc(sizeof(Packet));if (!packet) {perror("Failed to allocate memory");return NULL;}memcpy(&packet->header, ptr, sizeof(uint32_t));packet->header = ntohl(packet->header);ptr += sizeof(uint32_t);// 读取数据类型memcpy(&packet->type, ptr, sizeof(uint16_t));packet->type = ntohs(packet->type);ptr += sizeof(uint16_t);// 读取数据长度memcpy(&packet->length, ptr, sizeof(uint32_t));packet->length = ntohl(packet->length);ptr += sizeof(uint32_t);// 读取 CRC 校验值memcpy(&packet->crc, ptr, sizeof(uint32_t));packet->crc = ntohl(packet->crc);ptr += sizeof(uint32_t);// 读取数据主体packet = realloc(packet, sizeof(Packet) + packet->length);if (!packet) {perror("Failed to reallocate memory");return NULL;}memcpy(packet->payload, ptr, packet->length);return packet;
}

3. CRC 校验计算函数

用于计算数据的 CRC 校验值。

uint32_t calculateCRC(const char* data, size_t length) {uint32_t crc = 0xFFFFFFFF;for (size_t i = 0; i < length; i++) {crc ^= (uint32_t)data[i];for (int j = 0; j < 8; j++) {if (crc & 1)crc = (crc >> 1) ^ 0xEDB88320; // CRC-32 标准多项式elsecrc >>= 1;}}return crc ^ 0xFFFFFFFF;
}

4. TCP/UDP 适配

此协议可以通过 TCP 和 UDP 使用。以下是发送和接收示例:

  • TCP 通信

    // 使用 serializePacket 和 send() 发送数据
    size_t bufferSize;
    void* buffer = serializePacket(packet, &bufferSize);
    send(socket_fd, buffer, bufferSize, 0);
    free(buffer);// 接收数据后反序列化
    recv(socket_fd, recvBuffer, recvBufferSize, 0);
    Packet* receivedPacket = deserializePacket(recvBuffer);
    
  • UDP 通信

    // 使用 serializePacket 和 sendto() 发送数据
    size_t bufferSize;
    void* buffer = serializePacket(packet, &bufferSize);
    sendto(socket_fd, buffer, bufferSize, 0, (struct sockaddr*)&dest, sizeof(dest));
    free(buffer);// 接收数据后反序列化
    recvfrom(socket_fd, recvBuffer, recvBufferSize, 0, NULL, NULL);
    Packet* receivedPacket = deserializePacket(recvBuffer);
    

总结

  • 结构体封装了协议的核心字段。
  • 序列化与反序列化函数保证了数据可以在网络上传输和接收。
  • CRC 校验函数增强了数据的完整性校验。
  • TCP 和 UDP 的兼容性通过通用接口实现。此协议既适用于可靠的流式传输(TCP),也适用于快速的无连接传输(UDP)。

http://www.ppmy.cn/server/144813.html

相关文章

大数据实验4-HBase

一、实验目的 阐述HBase在Hadoop体系结构中的角色&#xff1b;能够掌握HBase的安装和配置方法熟练使用HBase操作常用的Shell命令&#xff1b; 二、实验要求 学习HBase的安装步骤&#xff0c;并掌握HBase的基本操作命令的使用&#xff1b; 三、实验平台 操作系统&#xff1…

【Docker集群应用】Docker基础与部署安装

文章目录 Docker概述Docker 与 虚拟机的区别Linux 六大命名空间Docker 的核心技术Docker 核心概念 Docker部署安装关闭防火墙和 SELinux安装依赖包设置阿里云镜像源安装 Docker-CE 并设置为开机自动启动查看 Docker 版本信息查看 Docker 信息Docker 系统信息 Docker 概述 Doc…

ctfshow-Misc入门(1-16)

misc1 查看图片得到flag misc2 1、打开文本&#xff0c;发现以“塒NG”开头 3、修改文件格式为png格式 4、查看图片&#xff0c;得到flag *遇到的问题&#xff1a;无法直接修改后缀名 *解决方法&#xff1a;需要点击文件夹&#xff0c;然后点击查看&#xff0c;将文件拓…

核心差异:知识VS文档管理(+工具软件安利)

在讨论知识管理和文档管理时&#xff0c;我们经常会听到这两种说法被混淆使用。然而&#xff0c;它们各自服务于不同的目的&#xff0c;这一点至关重要。 想象一下&#xff0c;你是一名项目经理&#xff0c;面临以下两项任务&#xff1a; 存储最新的项目计划捕捉团队讨论中获…

spring循环依赖以及MyBatis-Plus的继承特性导致循环依赖自动解决失效

在 Spring 中&#xff0c;循环依赖是指两个或多个 bean 之间相互依赖&#xff0c;形成一个循环引用的情况。Spring 对于循环依赖有一定的处理机制&#xff0c;但也存在一些限制。 一、Spring 处理循环依赖的方式 三级缓存解决构造函数注入的循环依赖&#xff1a; Spring 首先创…

C++结构型设计模式之使用抽象工厂来创建和配置桥接模式的例子

下面是一个使用抽象工厂模式来创建和配置桥接模式的示例&#xff0c;场景是创建不同操作系统的窗口&#xff08;Window&#xff09;及其对应的实现&#xff08;WindowImpl&#xff09;。我们将通过抽象工厂来创建不同操作系统下的窗口和实现。 代码示例 #include <iostrea…

机器学习周志华学习笔记-第5章<神经网络>

机器学习周志华学习笔记-第5章<神经网络> 卷王&#xff0c;请看目录 5模型的评估与选择5.1 神经元模型5.2 感知机与多层网络5.3 BP(误逆差)神经网络算法 5.4常见的神经网络5.4.1 RBF网络&#xff08;Radial Basis Function Network&#xff0c;径向基函数网络&#xff0…

移动光猫[HS8545M5-10]获取超密

移动光猫[HS8545M5-10]获取超级密码 1、缘由2、前期准备2.1、确保本地开通telnet客户端功能2.2、准备好相关软件 3、开始查找超密 1、缘由 最近想折腾一下ipv6ddns打通内外网&#xff0c;查询资料说是需要将光猫桥接到外网&#xff1b;但是使用光猫后边的用户名密码根本就找不到…