设计自己的网络通信协议
文章目录
- 一、为什么需要设计网络通信协议
- 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 算子,…...
C++初阶-list的底层
目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...
Spark 之 入门讲解详细版(1)
1、简介 1.1 Spark简介 Spark是加州大学伯克利分校AMP实验室(Algorithms, Machines, and People Lab)开发通用内存并行计算框架。Spark在2013年6月进入Apache成为孵化项目,8个月后成为Apache顶级项目,速度之快足见过人之处&…...
基于Flask实现的医疗保险欺诈识别监测模型
基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施,由雇主和个人按一定比例缴纳保险费,建立社会医疗保险基金,支付雇员医疗费用的一种医疗保险制度, 它是促进社会文明和进步的…...
基于服务器使用 apt 安装、配置 Nginx
🧾 一、查看可安装的 Nginx 版本 首先,你可以运行以下命令查看可用版本: apt-cache madison nginx-core输出示例: nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)
安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...
【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...
Bean 作用域有哪些?如何答出技术深度?
导语: Spring 面试绕不开 Bean 的作用域问题,这是面试官考察候选人对 Spring 框架理解深度的常见方式。本文将围绕“Spring 中的 Bean 作用域”展开,结合典型面试题及实战场景,帮你厘清重点,打破模板式回答,…...
MySQL体系架构解析(三):MySQL目录与启动配置全解析
MySQL中的目录和文件 bin目录 在 MySQL 的安装目录下有一个特别重要的 bin 目录,这个目录下存放着许多可执行文件。与其他系统的可执行文件类似,这些可执行文件都是与服务器和客户端程序相关的。 启动MySQL服务器程序 在 UNIX 系统中,用…...
