**STM32串口接收不定长数据原理与源程序**CSDN上有很多关于STM32串口接收不定长数据的文章,但实际使用后发现照搬他们的代码,程序根本就不能正确接收数据,其中最关键的一句有问题。其余内容完全正确。

文章末尾提供了源码链接,欢迎批评指正
*******以下文字基本参考原作者的内容,在此对其表示感谢,但对其中一个错误进行更改。码字不易希望能过,不然又有好多朋友走弯路

由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长字节的数据,由于STM32属于ARM单片机,所以这篇文章的方法也适合其他的ARM单片机。
IDLE中断什么时候发生?
IDLE就是串口收到一帧数据后,发生的中断。什么是一帧数据呢?比如说给单片机一次发来1个字节,或者一次发来8个字节,这些一次发来的数据,就称为一帧数据,也可以叫做一包数据。
如何判断一帧数据结束,就是我们今天讨论的问题。因为很多项目中都要用到这个,因为只有接收到一帧数据以后,你才可以判断这次收了几个字节和每个字节的内容是否符合协议要求。
看了前面IDLE中断的定义,你就会明白了,一帧数据结束后,就会产生IDLE中断。这个中断真是太TMD有用了。省去了好多判断的麻烦。
如何配置好IDLE中断?
下面我们就配置好串口IDLE中断吧。

这是串口CR1寄存器,其中,对bit4写1开启IDLE中断,对bit5写1开启接收数据中断。(注意:不同系列的STM32,对应的寄存器位可能不同)
(RXNE中断和IDLE中断的区别?
当接收到1个字节,就会产生RXNE中断,当接收到一帧数据,就会产生IDLE中断。比如给单片机一次性发送了8个字节,就会产生8次RXNE中断,1次IDLE中断。)


这是状态寄存器,当串口接收到数据时,bit5就会自动变成1,当接收完一帧数据后,bit4就会变成1.
需要注意的是,在中断函数里面,需要把对应的位清0,否则会影响下一次数据的接收。比如RXNE接收数据中断,只要把接收到的一个字节读出来,就会清除这个中断。IDLE中断,如何是F0系列的单片机,需要用ICR寄存器来清除,如果是F1系列的单片机,清除方法是“先读SR寄存器,再读DR寄存器”。(我怎么知道?手册上写的)
下面以STM32F103为例给出源程序(我用的是STM32F103ZET6,实测可行)。
我们先来看程序中的主要部分。
串口初始化函数片段

如果你原来的串口初始化函数具有打开串口接收中断的话,实际上就是在初始化函数中多了一条打开空闲中断的语句。

串口中断函数

口中断函数里面,最重要的两条语句,就是上图中圈出来的两条语句。第一条语句用来判断是否接收到1个字节,第二条语句用来判断是否接收到1帧数据。(是不是感觉超级方便?妈妈再也不用担心我如何判断是否接收完1帧数据了。)
主函数

我写的这个主函数,是用来验证接收的正确性的。RxCounter表示的是这一帧数据有几个字节,接收完一帧数据,会在中断函数里面把ReceiveState置1,然后,通过串口把接收到的数据发送回串口。这样,既验证了接收了多少字节的正确性,又验证了接收到的数据是否正确。

上图是结果验证。

以下是所有代码,供大家参考:
主函数main.c中的代码如下

#include "SysTick.h"
#include "led.h"
#include "usart.h"extern u8 receive_data[50];
extern u8 RXcounter;
extern u8 ReceiveState;int main()
{u8 i=0;  SysTick_Init(72);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);  //中断优先级分组LED_Init();USART1_Init(9600);while(1)                 //将接收到的数据发回串口助手{if(ReceiveState==1){u8 i=0;ReceiveState=0;while(RXcounter--){USART_SendData(USART1,receive_data[i++]);while(USART_GetFlagStatus(USART1,USART_FLAG_TC) != SET);}RXcounter=0;     //这句不能少,否则第二次接收数据时会出现丢失数据}}
}

usart.c中的代码如下

#include "usart.h" //包含头文件  u8 receive_data[50];       //定义全局变量
u8 RXcounter=0;                //定义全局变量
u8 ReceiveState=0;         //定义全局变量
/*******************************************************************************
USART1初始化函数
*******************************************************************************/
void USART1_Init(u32 bound)
{//GPIO端口设置GPIO_InitTypeDef GPIO_InitStructure;USART_InitTypeDef USART_InitStructure;NVIC_InitTypeDef NVIC_InitStructure;//打开对应时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);/*  配置GPIO的模式和端口 */GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;  //串口输出PA9GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;      //复用推挽输出GPIO_Init(GPIOA,&GPIO_InitStructure);  /*初始化串口输出IO */GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;   //串口输入PA10GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;       //浮空输入GPIO_Init(GPIOA,&GPIO_InitStructure);  /*初始化串口输出IO *///USART1初始化设置USART_InitStructure.USART_BaudRate = bound;//波特率设置USART_InitStructure.USART_WordLength = USART_WordLength_8b;//8位数据位USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位USART_InitStructure.USART_Parity = USART_Parity_No;//无校验USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;   //接收和发送模式USART_Init(USART1, &USART_InitStructure); //初始化串口1USART_Cmd(USART1, ENABLE);  //使能串口1USART_ClearFlag(USART1, USART_FLAG_TC);USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启接收中断USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);//开启总线空闲中断(一包数据发送完后,总线将处于空闲状态)//Usart1 NVIC 配置NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;//抢占优先级3NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;       //子优先级3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;         //IRQ通道使能NVIC_Init(&NVIC_InitStructure);    //根据指定的参数初始化NVIC寄存器
}/*******************************************************************************
串口中断函数
*******************************************************************************/
void USART1_IRQHandler(void)                    //串口1中断服务函数
{u8 Clear;if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)         //接收中断{receive_data[RXcounter++]=USART_ReceiveData(USART1);     //读取接收到的数据} if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)      //接收完1包数据后触发{Clear=USART1->SR;Clear=USART1->DR;ReceiveState=1;USART_ClearFlag(USART1,USART_FLAG_IDLE);}USART_ClearFlag(USART1,USART_FLAG_TC);
}

usart.h中的代码如下

#ifndef __usart_H
#define __usart_H#include "system.h" void USART1_Init(u32 bound);extern u8 receive_data[50];
extern u8 RXcounter;
extern u8 ReceiveState;#endif

源程序链接如下
链接:https://pan.baidu.com/s/19knjmhaI6sYtni6gNEpiPw
提取码:y6zo

如果你都已经看到了这里,给个赞再走吧!!!

STM32串口接收不定长数据原理与源程序相关推荐

  1. stm32串口接收不定长数据_基于STM32之UART串口通信协议--接收

    一.前言 1.简介 回顾上一篇UART发送当中,已经讲解了如何实现UART的发送操作了,接下来这一篇将会继续讲解如何实现UART的接收操作. 2.UART简介 嵌入式开发中,UART串口通信协议是我们 ...

  2. openmv串口数据 串口助手_STM32 串口接收不定长数据 STM32 USART空闲检测中断

    编者注: 单片机串口接收不定长数据时,必须面对的一个问题为:怎么判断这一包数据接收完成了呢?常见的方法主要有以下两种: 1.在接收数据时启动一个定时器,在指定时间间隔内没有接收到新数据,认为数据接收完 ...

  3. 基于HAL库STM32串口驱动不定长数据接收

    STM32串口驱动不定长数据接收带环形缓冲区 最新框架代码 使用方法 源码 串口接口文件 环形缓冲区接口文件 移植图示 使用涉及4个文件, UART_Port.c UART_Port.h Circul ...

  4. ZYNQ进阶之路14--PS端uart串口接收不定长数据

    ZYNQ进阶之路14--PS端uart串口接收不定长数据 导语 ZYNQ串口简介 实现步骤 导语 繁忙的博主又来了,本节我们实现一个比较简单的东西:串口.之前的章节中我们也有使用PS端的串口进行收发数 ...

  5. HAL库的串口基础学习(包含串口接收不定长数据的实现)

    HAL库的串口基础学习(1) HAL库有一个特点就是对于许多外设的初始化以及功能操作,都提供有一个weak版本的函数,这是充分的展现出库名字的含义(Hardware Abstraction Layer ...

  6. HAL库实践记录之串口接收不定长数据

    串口1接收不定长数据 实验板是原子mini板 一开始使用官方库,只能接受定长数据.把数据长度设置为1时,发送多字节数据时又会丢数.所以自己重写串口中断处理函数. 首先搞一下Cube配置用法:Mode选 ...

  7. 处理串口接收不定长数据的另一种解决方法

    开发平台:Keil 5 库函数版本:V3.5 芯片:STM32F103CBT6 之前我在我的另一篇博客中介绍过使用串口空闲中断+DMA的方式来处理不定长数据,没有看过的同学可以点击这里查看.今天要介绍 ...

  8. 串口接收不定长数据的几种方式

    在阅读本文前,你需要先做到串口成功接收一个数据(相信这一点是很简单的)  这几天简单总结了一下用串口怎么接收一帧数据的办法,个人使用的有三种,下面逐一介绍: 第一种:使用中断的方式: 这种在数据接收不 ...

  9. STM32使用串口IDLE中断的两种接收不定长数据的方式

    现在有很多数据处理都要用到不定长数据,而单片机串口的RXNE中断一次只能接收一个字节的数据,没有缓冲区,无法接收一帧多个数据,现提供两种利用串口IDLE空闲中断的方式接收一帧数据,方法如下: 方法1: ...

最新文章

  1. poj 3275(传递闭包)
  2. 从东岳流体下载自带OpenFOAM的Vmware虚拟机(Ubuntu20.04+OpenFOAM),无法共享文件夹【终极解决方案】
  3. ​使用高斯过程回归指导网络轻量化
  4. python 组合求和_39. 组合总和(Python)
  5. Linux命令之grep
  6. 交换机通过port-channel互联
  7. java 解析mp4文件头_视频文件头解析--MP4-获取mp4 文件信息
  8. faceframe kinect2.0 api 注意点
  9. 爬虫小程序 - 翻译君
  10. nowcoder20072 [HNOI2009]图的同构
  11. 抖音上热门的六大规律
  12. AiTrust下预训练和小样本学习在中文医疗信息处理挑战榜CBLUE表现
  13. nyoj 月老的难题【最大匹配】
  14. H323端口开放一览表
  15. 九天鸟p2p网贷系统的架构设计
  16. R语言学习 day_8
  17. mac下比较好用的svn软件,SVN客户端CornerStone 2.7.10 破解版
  18. 参加技术校园招聘,你最好先了解这五件事
  19. iOS 【地图绘制行政区域边界及填充】
  20. c语言 函数返回布尔值,返回值为布尔值

热门文章

  1. zufeoj 花生(The Peanuts)
  2. 虚幻4渲染编程(Pipeline篇)【第一卷:PBR Production Pipeline】
  3. python/keras中用Grid Search对神经网络超参数进行调参
  4. Linux component框架
  5. uni-app 104退出和解散群聊(二)
  6. python中定义函数时有用到箭头符号的,箭头主要干什么用
  7. 三星玄龙骑士电竞显示器Neo G9国内首秀,闪耀登陆2021ChinaJoy与UDE展
  8. win10系统克隆到新硬盘,如何克隆系统到新硬盘
  9. 好用的游戏机械键盘推荐什么?HyperX起源60键盘实力出圈
  10. [DV]闪存主控芯片验证全流程