串口数据包收发的思路和流程-stm32入门

news2025/1/10 1:58:39

本节主要内容:

  1. 如何去规定一个合理的数据包格式
  2. 如何收发数据包

1. 数据包格式规定/定义

1.1 HEX 数据包定义

  • 固定包长,含包头包尾
    在这里插入图片描述
  • 可变包长,含包头包尾
    在这里插入图片描述

首先数据包的作用是把一个个单独的数据给打包起来,方便我们进行多字节的数据通信。我们之前学习了串口的代码,发送一个字节,接收一个字节都没问题,但在实际应用中,我们可能需要把多个字节打包为一个整体进行发送,比如说,我们有个陀螺仪传感器,需要用串口发送数据到 STM32,陀螺仪的数据,比如 X 轴一个字节,Y 轴一个字节,Z 轴一个字节,总共 3 个数据,需要连续不断地发送,当你像这样,XYZXYZXYZ 连续发送的时候,就会出现一个问题,就是接收方,它不知道这数据哪个对应 X,哪个对应 Y,哪个对应 Z,因为接收方可能会从任意位置开始接收,所以会出现数据错位的现象,这时候,我们就需要研究一种方式,把这个数据进行分割,把 XYZ 这一批数据分隔开,分成一个个数据包,这样再接收的时候,就知道了数据包的第一个数据就是 X、第二个数据就是 Y,第三个数据就是 Z,这就是数据包的任务,就是把属于同一批的数据进行打包和分割,方便接收方进行识别,那有关分割打包的方法,可以自己发挥想象力来设计,只要逻辑行得通就行。

比如我可以设计,在这个 XYZXYZ 数据流中,数据包的第一个数据,也就是 X 的数据包,它的最高位置 1,其余数据包,最高位都置 0,当我接收到了数据之后,判断一下最高位,如果是 1,那就是 X 数据,然后紧跟着的两个数据就分别是 Y 和 Z,这就是一种可行的分割方法。这种方法就是把每个数据的最高位当作标志位来进行分割的,实际也有应用的例子,比如 UTF8 的编码方式,和这就是类似的,不过它那个编码更高级一些,感兴趣的话可以了解一下。

那本节我们主要讲的数据包分割方法,并不是在数据的高位添加标志位这种方式,因为这种方式破坏了原有数据,使用起来比较复杂,我们串口数据包,通常使用的是额外添加包头包尾的这种方式,比如我这里就列举了两种数据包格式,第一种是固定包长,含包头包尾,也就是每个数据包的长度都固定不变,数据包前面是包头,后面是包尾;第二种是可变包长,含包头包尾,也就是每个数据包的长度可以是不一样的,数据包前面是包头,后面是包尾。它们的数据包格式,可以是用户根据需求,自己规定的,也可以是你买一个模块,别的开发者规定的。

那本节规定是,比如固定包长这里,一批数据规定有 4 个字节,在这 4 个字节之前,加一个包头,比如我定义 0xFF 为包头,在 4 个字节之后,加一个包尾,比如我定义 0xFE 为包尾。那当我接收到 0xFF 之后,我就知道一个数据包来了,接着我再接收到的 4 个字节,就当作数据包的第 1、2、3、4 个数据,存在一个数组里,最后跟一个包尾,当我收到 0xFE 之后,就可以置一个标志位,告诉程序,我收到了一个数据包,然后新的数据包过来,再重复之前的过程,这样就可以在一个连续不断的数据流中,分割出我们想要的数据包了,这就是添加包头包尾实现数据分割打包的思路。

接着我们来研究几个问题:

  1. 包头包尾和数据载荷重复问题

这里定义 FF 为包头,FE 为包尾,如果我传输的数据本身就是 FF 和 FE 怎么办呢?那这个问题确实存在,如果数据和包头包尾重复,可能会引起误判。

对应这个问题我们有如下几种解决方法:

  1. 限制载荷数据的范围,如果可以的话,我们可以在发送的时候,对数据进行限幅,比如 XYZ,3 个数据,变化范围都可以是 0~100,那就好办了,我们可以在载荷中只发送 0~100 的数据,这样就不会和包头包尾重复了。
  2. 如果无法避免载荷数据和包头包尾重复,那我们就尽量使用固定长度的数据包,这样由于载荷数据是固定的,只要我们通过包头包尾对齐了数据,我们就可以严格知道,哪个数据应该是包头包尾,哪个数据应该是载荷数据,在接收载荷数据的时候,我们并不会判断它是否是包头包尾,而在接收包头包尾的时候,我们会判断它是不是确实是包头包尾,用于数据对齐。这样,在经过几个数据包的对齐之后,剩下的数据包应该就不会出现问题了。
  3. 增加包头包尾的数量,并且让它尽量呈现出载荷数据出现不了的状态,比如我们使用 FF、FE 做为包头,FD、FC 作为包尾,这样也可以避免载荷数据和包头包尾重复的情况发生。
  1. 包头包尾并不是全部都需要的。

比如我们可以只要一个包头,把包尾删掉,这样数据包的格式就是,一个包头 FF,加 4 个数据,这样也是可以的,当检测到 FF,开始接收,收够 4 个字节后,置标志位,一个数据包接收完成,这样也可以,不过这样的话,载荷和包头重复的问题会更严重一些,比如最严重的情况下,我载荷全是 FF,包头也是 FF,那你肯定不知道哪个是包头了,而加上 FE 作为包尾,无论数据怎么变化都是可以分辨出包头包尾的。

  1. 固定包长和可变包长的选择问题

对于 HEX 数据包来说,如果你的载荷会出现和包头包尾重复的情况,那就最好选择固定包长,这样可以避免接收错误,如果你又会重复,又选择可变包长,那数据很容易就乱套了;如果载荷不会和包头包尾重复,那可以选择可变包长,数据长度,像这样,4 位、3 位、等等,1 位、10 位,来回任意变肯定都没问题。因为包头包尾是唯一的,只要出现包头,就开始数据包,只要出现包尾,就结束数据包,这样就非常灵活了。

  1. 各种数据转换为字节流的问题

这里数据包都是一个字节一个字节组成的,如果你想发送 16 位的整型数据、32 位的整型数据、float、double、甚至是结构体,其实都没问题,因为它们内部其实都是由一个字节一个字节组成的,只需要用一个 uint8_t 的指针指向它,把它们当作一个字节数组发送就行了。

1.2 文本数据包定义

  • 固定包长,含包头包尾
    在这里插入图片描述
  • 可变包长,含包头包尾
    在这里插入图片描述

文本数据包和 HEX 数据包,就分别对应了文本模式和 HEX 这两种模式。在 HEX 数据包里面,数据都是以原始的字节数据本身呈现的,而在文本数据包里面,每个字节就经过了一层编码和译码,最终表现出来的,就是文本格式,但实际上,每个文本字符背后,其实都还是一个字节的 HEX 数据。

那我们看一下,同样给出了固定包长和可变包长这两种模式,由于数据译码成了字符形式,这就会存在大量的字符可以作为包头包尾,可以有效避免载荷和包头包尾重复的问题。比如本节规定以 @ 这个字符作为包头,以 \r\n 也就是换行,这两个字符作为包尾,在载荷数据中间可以出现除了包头包尾的任意字符,这很容易做到,所以文本数据包基本不用担心载荷和包头包尾重复的问题,使用非常灵活。可变包长、各种字母、符号、数字,都可以随意使用,当我们接收到载荷数据之后,得到的就是一个字符串,在软件中再对字符串进行操作和判断,就可以实现各种指令控制的功能了。而且字符串数据包表达的意义很明显,可以把字符串数据包直接打印到串口助手上,什么指令,什么数据,一眼就能看明白,所以这个文本数据包,通常会以换行作为包尾,这样在打印的时候,就可以一行一行地显示了,非常方便。

那 HEX 数据包与文本数据包这两种对比下来,其实也是各有优缺点的。
HEX 数据包:

  • 优点:传输最直接,解析数据非常简单,比较适合一些模块发送原始的数据。比如一些使用串口通信的陀螺仪、温湿度传感器。
  • 缺点:灵活性不足、载荷容易和包头包尾重复。

文本数据包:

  • 优点:数据直观易理解,非常灵活,比较适合一些输入指令进行人机交互的场合,比如蓝牙模块常用的 AT 指令,CNC 和 3D 打印机常用的 G 代码,都是文本数据包的格式。
  • 缺点:解析效率低。比如发送一个数 100,HEX 数据包就是一个字节 100,完事,文本数据包就得是 3 个字节的字符,‘1’,‘0’,‘0’,收到之后还要把字符转换成数据,才能得到 100。所以说,我们需要根据实际场景来选择和设计数据包格式。

2. 数据包的收发流程

2.1 数据包的发送

数据包的发送非常简单。

在 HEX 数据包这里,我如果想发送一个数据包,就定义一个数组,填充数据,然后用上节我们写过的 SendArray 一发就完事了。

在 文本 数据包这里,我如果想发送数据包,就写一个字符串,然后调用 SendString 一发送也完事了。

数据包的发送非常简单。因为发送过程是完全自主可控的,想发啥就发啥。我们写到的时候也能感受到,串口发送,比接收简单多了。

2.2 数据包的接收

接受一个数据包就比较复杂了。我们来学习一下。
在这里插入图片描述

这里演示了固定包长 HEX 数据包的接收方法。首先,根据前一节的代码,我们知道,每收到一个字节,程序都会进一遍中断,在中断函数里,我们可以拿到这一个字节,但拿到之后,我们就得退出中断了,所以,每拿到一个数据,都是一个独立的过程,而对于数据包来说,很明显它具有前后关联性,包头之后是数据,数据之后是包尾,对于包头、数据和包尾这三种状态,我们都需要有不同的处理逻辑,所以在程序中,我们需要设计一个能记住不同状态的机制,在不同状态执行不同的操作,同时还要进行状态的合理转移,这种程序设计思维,就叫做“状态机”。

在这里我们就使用状态机的方法来接收一个数据包,要想设计一个好的状态机程序,画一个下面这样的状态转移图是必要的。对于上面这样一个固定包长 HEX 数据包来说,我们可以定义 3 个状态,第一个状态是等待包头,第二个状态是接收数据,第三个状态是等待包尾。每个状态需要用一个变量来标志一下,比如我这里用变量 S 来标志,三个状态依次为 S = 0,S = 1,S = 2,这一点类似于置标志位,只不过标志位只有 0 和 1,而状态机是多标志位状态的一种方式,然后执行流程是:

  1. 最开始,S = 0,收到一个数据,进中断,根据 S = 0,进入第一个状态的程序,判断数据是不是包头 FF,如果是 FF,则代表收到包头,之后置 S = 1,退出中断,结束,这样下次再进中断,根据 S = 1,就可以接收数据的程序了,那在第一个状态,如果收到的不是 FF,就证明数据包没有对齐,我们应该等待数据包包头的出现,这时状态就仍然是 0,下次进中断,就还是判断包头的逻辑,直到出现 FF,才能转到下一个状态。
  2. 那之后,出现了 FF,我们就可以转移到接收数据的状态了。这时再收到数据,我们就直接把它存在数组里,另外再用一个变量,记录收了多少个数据,如果没收够四个数据,就一直是接收状态,如果收够了,就置 S = 2,下次中断时,就可以进入下一个状态了。
  3. 那最后一个状态就是等待包尾了,判断数据是不是 FE,正常情况,应该是 FE,这样就可以置 S = 0,回到最初的状态,开始下一个轮回,当然也有可能这个数据不是 FE,比如数据和包头重复了,导致包头位置判断错了,那这个包尾位置就有可能不是 FE,这时就可以进入重复等待包尾的状态,直到接收到真正的包尾。这样加入包尾的判断,更能预防因数据和包头重复造成的错误。

这就是使用状态机接收数据包的思路。这个状态机其实是一种很广泛的编程思路,在很多地方都可以用到。使用的基本步骤是:

  1. 先根据项目要求定义状态,画几个圈
  2. 然后考虑好各个状态在什么情况下会进行转移,如何转移,画好线和转移条件。
  3. 最后根据这个图来进行编程。

这样思维就会非常清晰了。比如你要做个菜单,就可以用到状态机的思维,按什么键,切换什么菜单,执行什么样的程序,还有一些芯片内部逻辑,也会用到状态机。比如芯片什么情况下进入待机状态,什么情况下进入工作状态,这也是状态机的应用,希望大家可以研究一下,对你的编程肯定会有帮助。

在这里插入图片描述

这里演示了可变包长 文本 数据包的接收方法。我们看一下可变包长 文本 数据包的接收流程,同样也是利用状态机,定义了三个状态。

  1. 第一个状态,等待包头,判断收到的是不是我们规定的 @ 符号,如果收到 @,就进入接收状态。
  2. 在接收状态下,依次接收数据,同时,这个状态还应该要兼具等待包尾的功能,因为这是可变包长,我们接收数据的时候,也要时刻监视,是不是收到包尾了,一旦收到包尾了,就结束,那这里,这个状态的逻辑就应该是,收到一个数据,判断是不是 \r,如果不是,就正常接收,如果是,则不接收,同时跳到下一个状态,等待包尾 \n,因为这里数据包有两个包尾 \r,\n,所以需要第三个状态。
  3. 如果只有一个包尾,那在出现包尾之后,就可以直接回到初始状态了,只需要两个状态就行。因为接收数据和等待包尾需要在一个状态里同时进行。由于串口的包头包尾不会出现在数据中,所以基本不会出现数据错位的现象。

这就是使用状态机接收文本数据包的方法。

其他的数据包也都可以套用这两个形式,等会我们写程序就会根据这里面的流程来。

到这里,关于数据包的定义、分类、优缺点和注意事项就介绍完了。

包头 FF 和 包尾 FE 用于控制接收,没有显示。

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

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

相关文章

Linux git

1.Git 初识 不知道你⼯作或学习时,有没有遇到这样的情况:我们在编写各种⽂档时,为了防止文档丢失,更改失误,失误后能恢复到原来的版本,不得不复制出⼀个副本,⽐如: “报告-v1”? …

python爬虫指南之请求模块urllib的详细教程

文章目录 前言一、urllib的子模块二、HttpResponse常用方法与属性获取信息urlli.parse的使用(一般用于处理带中文的url) 三、爬取baidu官网HTML源代码添加请求头信息(重构user\_agent) 四、扩展知识with open和open两者的区别关于Python技术储备一、Pyth…

带你用uniapp从零开发一个仿小米商场_10. 首页开发

图标菜单栏开发 轮播图开发完成后,就是图标菜单栏了 可以看出这些图标都是一样的样式,所以可以勇哥flex布局让他们每个占百分之20 代码如下,既然都是一样的那就直接用个循环嵌套一下 data数据如下 同样,为了能让这段代码能在别的地方也用到,我直接把它封装成组件 <templ…

不常在港居住!香港高才通计划续签5大方式的利弊汇总!

不常在港居住&#xff01;香港高才通计划续签5大方式的利弊汇总&#xff01; 今年香港高才通计划申请真的蛮火的&#xff01;不过申请高才计划成功后续签问题也难倒了不少人&#xff0c;那么应该怎样准备高才续签呢&#xff1f;今天就仔细说说&#xff01; 高才通的逗留模式&am…

uniapp挽留提示2.0

项目需求&#xff1a;有时候挽留的ui是全屏的&#xff0c;用page-container也可以。后来产品提了个问题&#xff0c;手机侧滑的时候没那么顺畅&#xff08;就是一用侧滑&#xff0c;就显示出来&#xff0c;产品要的方案是如下图&#xff0c;emmm大概是这个意思&#xff09; 后面…

【javaWeb】HTTP协议

HTTP (全称为 “超文本传输协议”) 是一种应用非常广泛的应用层协议 HTTP 是一个文本格式的协议. 可以通过 Chrome 开发者工具或者 Fiddler 抓包, 分析 HTTP 请求/响应的细节. 上图是通过Fiddler对访问百度搜索页时抓取的一个http协议的包。 观察抓包结果,可以看到,当前 http…

数据分析实战案例:Python 分析员工为何离职(附完整代码)

大家好&#xff0c;今天给大家介绍一个Python数据分析项目实战&#xff0c;不仅包含代码&#xff0c;还提供分析数据集。 员工流失或是员工离开公司的比率是公司关注的一个重要问题。它不仅会导致宝贵人才的流失&#xff0c;还会产生成本并破坏生产力。了解员工辞职的原因对于…

中电金信鲸Bot RPA荣获最佳人工智能解决方案

近年来&#xff0c;数字经济已成为国家“十四五”规划和“新基建”战略的重要支撑。银行业作为我国经济体系的重要组成部分&#xff0c;其发展战略也出现了新的变化。数字化智能化转型成为银行业新的利润增长点&#xff0c;科学制定实施数字化转型战略成为下一步数字化转型的首…

【软件测试】盘一盘工作中遇到的 MQ 异常测试

上一篇小结了一下关于redis的异常测试&#xff0c;今天再来盘一盘 MQ 相关的。MQ 跟 redis 一样&#xff0c;也是现在系统服务中不可或缺的重要中间件&#xff0c;通常用来流量削峰、应用解耦、异步处理等。 日常经手的系统主要用的是 RocketMQ&#xff0c;是阿里系下开源的一…

cmdline

cmdline是一个kv结构,就是uboot参数传给kernel使用的 举例: Kernel command line: user_debug=31 storagemedia=mtd androidboot.storagemedia=mtd androidboot.mode=normal mac=00FA89112233 serial=LONBON12345 earlycon=uart8250,mmio32,0xff570000 console=ttyFIQ0…

如何使用JMeter测试https请求

HTTP与HTTPS略有不同&#xff0c;所以第一次使用JMeter测试https请求时遇到了问题&#xff0c;百度一番后找到解决方法&#xff1a;加载证书。 下面内容主要记录这次操作&#xff0c;便于后续参考&#xff1a; 操作浏览器&#xff1a;谷歌 &#xff08;1&#xff09;下载被测…

122.买卖股票的最佳时机II(不限次数)

题目 题解 labuladong的状态图解 class Solution:def maxProfit(self, prices: List[int]) -> int:N len(prices)# 定义状态&#xff1a;dp[i][j]表示在第i天持有或卖出时的最大利润&#xff0c;j1代表持有&#xff0c;j0代表卖出dp [[0 for j in range(2)] for i in ra…

MySQL学习day03

一、SQL图形化界面工具 常用比较常用的图形化界面有sqlyog、mavicat、datagrip datagrip工具使用相当方便&#xff0c;功能比前面两种都要强大。 DataGrip工具的安装和使用请查看这篇文档&#xff1a;DataGrip 安装教程 DML-介绍 DML全称是Data Manipulation Language(数据…

视频制作技巧:背景图片与视频画中画效果的完美结合

在视频制作过程中&#xff0c;背景图片和画中画效果是常用的技术&#xff0c;它们可以极大地增强视频的视觉效果和表现力。可以制作出更加独特、吸引的视频作品。现在一起看下云炫AI智剪如何批量制作画中画的操作吧。 一、选择合适的背景图片 在视频制作中&#xff0c;选择合…

WiseAlign 软件运行中存图功能使用方法

WiseAlign 软件运行中存图功能使用方法 在需要存图的相机图像通道点击鼠标右键 在弹出的菜单中选择“图像操作——保存图像” 选择想要存放图片的文件夹&#xff08;如下图所示&#xff09; 修改文件名称 如果文件夹中已有同名文件会提示xxx.bmp文件已存在&#xff0c;是否需要…

Unity EventSystem的一些理解和使用

Unity的EventSystem是用于处理用户输入和交互的系统。它是Unity UI系统的核心组件之一&#xff0c;可以用于捕捉和分发各种事件&#xff0c;例如点击、拖拽、按键、射线等。 常用的属性和方法有以下这些&#xff1a; 属性&#xff1a; current: 获取当前的EventSystem实例。…

企业被列入经营异常会有什么后果?

1、工商方面的不良影响 被工商纳入异常吊销营业执照&#xff1a;公司地址异常将会被工商部门纳入经营异常名录&#xff0c;需要及时申请移出&#xff0c;否则会影响正常经营&#xff0c;严重则有被吊销营业执照的风险。 影响企业形象及信誉度&#xff1a;企业间的合作都非常重视…

Linux 内核栈保护

栈保护可以检测栈被写坏的情况。如果怀疑有此类情况&#xff0c;可以将栈保护打开试试 详细可参考文章 栈保护杂记-CSDN博客 栈保护开启 下图是关于strong的解释&#xff0c;在什么情况下会加入栈保护 后面由于gcc版本的原因&#xff0c;好像不支持开启栈保护。后面再进行效果…

数据结构和算法-树和二叉树的定义和基本术语和性质

文章目录 树的基本概念和相关术语相关的应用节点间的关系描述节点&#xff0c;树的属性描述有序树vs无序树树vs森林小结 树的相关性质考点1考点2考点3考点4考点5考点6小结 二叉树的相关概念和基本术语重要 &#xff08;五种状态&#xff09;特殊二叉树小结 二叉树的相关性质二叉…

Java - Stream Filter 多条件筛选过滤

Java Stream流中Filter用于通过设置的条件过滤出元素 &#xff0c;示例如下&#xff1a; List strings Arrays.asList(“abc”, “”, “bc”, “efg”, “abcd”,"", “jkl”);List filtered strings.stream().filter(string -> !string.isEmpty()).collect(C…