设计自己的网络通信协议
文章目录
- 一、为什么需要设计网络通信协议
- 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 算子,…...
日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
进程地址空间(比特课总结)
一、进程地址空间 1. 环境变量 1 )⽤户级环境变量与系统级环境变量 全局属性:环境变量具有全局属性,会被⼦进程继承。例如当bash启动⼦进程时,环 境变量会⾃动传递给⼦进程。 本地变量限制:本地变量只在当前进程(ba…...
VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
Java 加密常用的各种算法及其选择
在数字化时代,数据安全至关重要,Java 作为广泛应用的编程语言,提供了丰富的加密算法来保障数据的保密性、完整性和真实性。了解这些常用加密算法及其适用场景,有助于开发者在不同的业务需求中做出正确的选择。 一、对称加密算法…...
令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
全志A40i android7.1 调试信息打印串口由uart0改为uart3
一,概述 1. 目的 将调试信息打印串口由uart0改为uart3。 2. 版本信息 Uboot版本:2014.07; Kernel版本:Linux-3.10; 二,Uboot 1. sys_config.fex改动 使能uart3(TX:PH00 RX:PH01),并让boo…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
安宝特案例丨Vuzix AR智能眼镜集成专业软件,助力卢森堡医院药房转型,赢得辉瑞创新奖
在Vuzix M400 AR智能眼镜的助力下,卢森堡罗伯特舒曼医院(the Robert Schuman Hospitals, HRS)凭借在无菌制剂生产流程中引入增强现实技术(AR)创新项目,荣获了2024年6月7日由卢森堡医院药剂师协会࿰…...
【SpringBoot自动化部署】
SpringBoot自动化部署方法 使用Jenkins进行持续集成与部署 Jenkins是最常用的自动化部署工具之一,能够实现代码拉取、构建、测试和部署的全流程自动化。 配置Jenkins任务时,需要添加Git仓库地址和凭证,设置构建触发器(如GitHub…...
