设计自己的网络通信协议
文章目录
- 一、为什么需要设计网络通信协议
- 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 校验值。协议使用结构体封装,并设计了适用于 TCP 和 UDP 的传输方式。
通信协议设计
协议字段描述
字段名称 | 类型 | 描述 |
---|---|---|
Header | uint32_t | 消息头(标识协议类型或版本,固定值)。 |
Type | uint16_t | 数据类型(如文本、文件、命令等)。 |
Length | uint32_t | 数据主体的长度(单位:字节)。 |
CRC | uint32_t | CRC 校验值,用于校验数据完整性。 |
Payload | char[] | 数据主体(可变长度)。 |
数据结构定义
#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)。
相关文章:
设计自己的网络通信协议
文章目录 一、为什么需要设计网络通信协议1. **标准化通信规则**2. **确保数据传输的可靠性**3. **支持网络的多样性和可扩展性**4. **分层设计,简化复杂性**5. **实现设备的互操作性**6. **支持多任务和多应用并发**7. **提供安全性**8. **支持不同的通信模式**总结…...
深入理解 Seata:分布式事务的最佳解决方案
随着微服务架构的广泛应用,分布式事务管理成为系统设计中一项重要且极具挑战的任务。在微服务架构下,服务之间通过网络调用,单个业务操作往往需要多个服务的协作来完成,这样分布式事务的问题就不可避免。Seata 是目前较为流行的一…...
JDK下载
jdk-8u421-windows-x64.exe : 阿里云盘 jdk-7u80-windows-x64.exe :阿里云盘...

如何使用 Python 开发一个简单的文本数据转换为 Excel 工具
目录 一、准备工作 二、理解文本数据格式 三、开发文本数据转换为Excel工具 读取CSV文件 将DataFrame写入Excel文件 处理其他格式的文本数据 读取纯文本文件: 读取TSV文件: 四、完整代码与工具封装 五、使用工具 六、总结 在数据分析和处理的日常工作中,我们经常…...

React(六)——Redux
文章目录 项目地址基本理解一、配置Redux store二、创建slice配置到store里并使用三、给Slice配置reducers,用来修改初始值 项目地址 教程作者:教程地址: 代码仓库地址: 所用到的框架和插件: dbt airflow基本理解 s…...

java抽奖系统(二)
3. 新建项目 3.1 选择相应的框架 pom文件配置如下: <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:s…...

STM32F10x 定时器
使用定时器实现:B5 E5的开关 添加相关的.h路径文件 添加相关的.c配置文件 led.h文件 用于声明LED函数 #ifndef __LED_H //没有定义__LED_H #define __LED_H //就定义__LED_H #define LED1_ON GPIO_ResetBits(GPIOB,GPIO_Pin_5) #defi…...

从0开始学PHP面向对象内容之常用设计模式(适配器,桥接,装饰器)
二,结构型设计模式 上两期咱们讲了创建型设计模式,都有 单例模式,工厂模式,抽象工厂模式,建造者模式,原型模式五个设计模式。 这期咱们讲结构型设计模式 1、适配器模式(Adapter) …...

玩转数字与运算:用C语言实现24点游戏的扑克牌魅力
✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:Java Fans的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏…...

前端入门之VUE--基础与核心
前言 VUE是前端用的最多的框架;这篇文章是本人大一上学习前端的笔记;欢迎点赞 收藏 关注,本人将会持续更新。 Vue学习笔记 用于构建用户界面的渐进式框架 构建用户界面:基于数据动态渲染页面渐进式:循序渐近的学…...

logback 初探学习
logback 三大模块 记录器(Logger)、追加器(Appender)和布局(Layout) 配置文件外层最基本的标签如图示 xml中定义的就是这个三个东西下面进入学习 包引入参考springboot 官方文档 Logging :: Spring Boo…...

在Elasticsearch中,是怎么根据一个词找到对应的倒排索引的?
大家好,我是锋哥。今天分享关于【在Elasticsearch中,是怎么根据一个词找到对应的倒排索引的?】面试题。希望对大家有帮助; 在Elasticsearch中,是怎么根据一个词找到对应的倒排索引的? 在 Elasticsearch 中…...

1992-2021年 各省市县经过矫正的夜间灯光数据(GNLD、VIIRS)区域汇总:省份、城市、区县面板数据
1992-2021年 各省市县经过矫正的夜间灯光数据(GNLD、VIIRS)区域汇总:省份、城市、区县面板数据 .r.rar https://download.csdn.net/download/2401_84585615/90001905 从1992年至2021年,中国各省份、城市及区县的夜间灯光数据经过…...

linux实战-黑链——玄机靶场
黑链的特征: 隐藏链接:黑链通常隐藏在网站页面中,使用CSS、JavaScript或其他手段使其对普通用户不可见,但仍然能被搜索引擎爬虫检测到。恶意内容:这些链接指向的内容可能包含恶意软件、钓鱼页面或其他不良内容&#x…...

鸿蒙NEXT开发案例:字数统计
【引言】 本文将通过一个具体的案例——“字数统计”组件,来探讨如何在鸿蒙NEXT框架下实现这一功能。此组件不仅能够统计用户输入文本中的汉字、中文标点、数字、以及英文字符的数量,还具有良好的用户界面设计,使用户能够直观地了解输入文本…...

uniapp vue2项目迁移vue3项目
uniapp vue2项目迁移vue3项目,必须适配的部分 一、main.js 创建应用实例 // 之前 - Vue 2 import Vue from vue import App from ./App Vue.config.productionTip false // vue3 不再需要 App.mpType app // vue3 不再需要 const app new Vue({ ...App }) …...

16.C++STL 3(string类的模拟,深浅拷贝问题)
⭐本篇重点:string类的模拟,自己实现一个简单的string类 ⭐本篇代码:c学习/05.string类的学习 橘子真甜/c-learning-of-yzc - 码云 - 开源中国 (gitee.com) 目录 一. 经典string类的模拟 1.1 深浅拷贝问题 1.2 使用深拷贝完成经典string类的…...
神经网络10-Temporal Fusion Transformer (TFT)
Temporal Fusion Transformer (TFT) 是一种专为时序数据建模而设计的深度学习模型,它结合了Transformer架构和其他技术,旨在有效地处理和预测时序数据中的复杂模式。TFT 于 2020 年由 Google Research 提出,旨在解决传统模型在时序预测中的一…...

“iOS profile文件与私钥证书文件不匹配”总结打ipa包出现的问题
目录 文件和证书未加载或特殊字符问题 证书过期或Profile文件错误 确认开发者证书和私钥是否匹配 创建证书选择错误问题 申请苹果 AppId时勾选服务不全问题 总结 在上线ios平台的时候,在Hbuilder中打包遇见了问题,生成ipa文件时候,一…...

《图像梯度与常见算子全解析:原理、用法及效果展示》
简介:本文深入探讨图像梯度相关知识,详细介绍图像梯度是像素灰度值在不同方向的变化速度,并以 “pig.JPG” 图像为例,通过代码展示如何选取图像部分区域并分析其像素值以论证图像梯度与边缘信息的关联。接着全面阐述了 Sobel 算子,…...

接口测试中缓存处理策略
在接口测试中,缓存处理策略是一个关键环节,直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性,避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明: 一、缓存处理的核…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...
在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用
1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...

优选算法第十二讲:队列 + 宽搜 优先级队列
优选算法第十二讲:队列 宽搜 && 优先级队列 1.N叉树的层序遍历2.二叉树的锯齿型层序遍历3.二叉树最大宽度4.在每个树行中找最大值5.优先级队列 -- 最后一块石头的重量6.数据流中的第K大元素7.前K个高频单词8.数据流的中位数 1.N叉树的层序遍历 2.二叉树的锯…...

算法岗面试经验分享-大模型篇
文章目录 A 基础语言模型A.1 TransformerA.2 Bert B 大语言模型结构B.1 GPTB.2 LLamaB.3 ChatGLMB.4 Qwen C 大语言模型微调C.1 Fine-tuningC.2 Adapter-tuningC.3 Prefix-tuningC.4 P-tuningC.5 LoRA A 基础语言模型 A.1 Transformer (1)资源 论文&a…...

七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...

无人机侦测与反制技术的进展与应用
国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机(无人驾驶飞行器,UAV)技术的快速发展,其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统,无人机的“黑飞”&…...

uniapp手机号一键登录保姆级教程(包含前端和后端)
目录 前置条件创建uniapp项目并关联uniClound云空间开启一键登录模块并开通一键登录服务编写云函数并上传部署获取手机号流程(第一种) 前端直接调用云函数获取手机号(第三种)后台调用云函数获取手机号 错误码常见问题 前置条件 手机安装有sim卡手机开启…...