news 2026/4/23 2:44:01

STM32 USART接收中断:如何判断数据接收完成?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
STM32 USART接收中断:如何判断数据接收完成?

引言

在STM32的串口通信开发中,中断方式接收数据是最常见的方式之一。然而,很多开发者都会遇到一个关键问题:"如何判断一帧数据已经接收完成?"今天我们就来深入探讨这个问题,并提供几种实用的解决方案。

一、为什么需要判断数据接收完成?

在串口通信中,数据是以字节流的形式传输的。当我们在中断服务函数中每次只接收一个字节时,需要一种机制来判断当前接收的数据是否构成一个完整的消息帧。常见的应用场景包括:

  • 接收不定长数据帧

  • 解析协议数据包(如Modbus、自定义协议等)

  • 处理命令行指令

  • 接收传感器数据

二、基本原理

在USART接收中断服务函数中,我们通常会这样开始:

void USART1_IRQHandler(void) { if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { // 读取接收到的数据 uint8_t rx_data = USART_ReceiveData(USART1); // ... 处理数据 } }

但这里只接收了一个字节,如何知道数据接收完成了呢?

三、四种判断数据接收完成的方法

方法1:超时判断法

这是最常用的方法之一,通过判断相邻两个字节之间的时间间隔来判断数据是否接收完成。

// 定义接收结构体 typedef struct { uint8_t buffer[256]; uint16_t index; uint8_t flag; uint32_t last_time; } UART_RxTypeDef; UART_RxTypeDef uart1_rx; // 中断服务函数 void USART1_IRQHandler(void) { if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); // 记录当前时间(可以使用SysTick或定时器) uart1_rx.last_time = SysTick->VAL; // 存储数据 if(uart1_rx.index < 256) { uart1_rx.buffer[uart1_rx.index++] = data; } USART_ClearITPendingBit(USART1, USART_IT_RXNE); } } // 主循环或定时器中断中检查超时 void Check_UART_Timeout(void) { uint32_t current_time = SysTick->VAL; uint32_t time_diff = abs(current_time - uart1_rx.last_time); // 如果超过设定的超时时间(如10ms) if(time_diff > 10000 && uart1_rx.index > 0) // 10ms超时 { uart1_rx.flag = 1; // 标记数据接收完成 } }

优点

  • 适用于不定长数据

  • 实现相对简单

缺点

  • 需要额外的定时器资源

  • 超时时间需要根据波特率调整

方法2:特定帧头帧尾法

这种方法适用于有固定格式的协议。

#define FRAME_HEADER 0xAA #define FRAME_FOOTER 0x55 typedef enum { RX_STATE_IDLE, RX_STATE_HEADER, RX_STATE_DATA, RX_STATE_COMPLETE } RxStateTypeDef; void USART1_IRQHandler(void) { static RxStateTypeDef rx_state = RX_STATE_IDLE; static uint8_t rx_index = 0; static uint8_t rx_buffer[256]; if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); switch(rx_state) { case RX_STATE_IDLE: if(data == FRAME_HEADER) { rx_index = 0; rx_state = RX_STATE_HEADER; } break; case RX_STATE_HEADER: rx_buffer[rx_index++] = data; rx_state = RX_STATE_DATA; break; case RX_STATE_DATA: if(data == FRAME_FOOTER) { rx_state = RX_STATE_COMPLETE; // 数据接收完成处理 Process_Complete_Frame(rx_buffer, rx_index); } else if(rx_index < 255) { rx_buffer[rx_index++] = data; } else { // 缓冲区溢出,重置状态 rx_state = RX_STATE_IDLE; } break; default: rx_state = RX_STATE_IDLE; break; } USART_ClearITPendingBit(USART1, USART_IT_RXNE); } }

优点

  • 可靠性高

  • 适合有固定格式的协议

缺点

  • 数据中不能出现与帧头帧尾相同的字符

  • 需要转义机制或使用字节填充

方法3:固定长度法

如果数据长度是固定的,这种方法最简单。

#define FIXED_LENGTH 10 void USART1_IRQHandler(void) { static uint8_t rx_buffer[FIXED_LENGTH]; static uint8_t rx_count = 0; if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { rx_buffer[rx_count++] = USART_ReceiveData(USART1); if(rx_count >= FIXED_LENGTH) { // 数据接收完成 Process_Complete_Frame(rx_buffer, FIXED_LENGTH); rx_count = 0; } USART_ClearITPendingBit(USART1, USART_IT_RXNE); } }

优点

  • 实现简单

  • 效率高

缺点

  • 只适用于固定长度数据

  • 缺乏灵活性

方法4:长度字段法

在数据包中包含长度信息,这是最专业的方法。

typedef struct { uint8_t header; // 帧头 uint8_t length; // 数据长度 uint8_t data[255]; // 数据 uint8_t checksum; // 校验和 } UART_FrameTypeDef; void USART1_IRQHandler(void) { static uint8_t rx_state = 0; static uint8_t rx_length = 0; static uint8_t rx_count = 0; static uint8_t rx_buffer[256]; if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); switch(rx_state) { case 0: // 等待帧头 if(data == 0xAA) { rx_state = 1; rx_count = 0; } break; case 1: // 获取长度 rx_length = data; if(rx_length > 0 && rx_length <= 255) { rx_state = 2; } else { rx_state = 0; // 长度错误,重新开始 } break; case 2: // 接收数据 rx_buffer[rx_count++] = data; if(rx_count >= rx_length) { rx_state = 3; } break; case 3: // 接收校验和 if(Verify_Checksum(rx_buffer, rx_length, data)) { // 数据接收完成且校验通过 Process_Complete_Frame(rx_buffer, rx_length); } rx_state = 0; break; } USART_ClearITPendingBit(USART1, USART_IT_RXNE); } }

优点

  • 灵活,支持变长数据

  • 可靠性高

缺点

  • 实现相对复杂

  • 需要处理异常情况

四、实战建议

1.结合使用多种方法

在实际项目中,我推荐结合使用超时判断和协议解析。例如:

  • 使用超时机制作为安全保障

  • 使用协议解析作为主要判断依据

2.使用DMA+IDLE中断(高级方法)

对于STM32的高端型号,可以使用DMA配合IDLE中断,这是最高效的方法:

// 启用IDLE中断 USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); void USART1_IRQHandler(void) { // 接收中断 if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { // DMA自动接收,无需在此处理 USART_ClearITPendingBit(USART1, USART_IT_RXNE); } // IDLE中断 - 检测到总线空闲 if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET) { // 清除IDLE中断标志(先读USART_SR,再读USART_DR) volatile uint32_t temp = USART1->SR; temp = USART1->DR; (void)temp; // 获取DMA接收的数据长度 uint16_t rx_len = BUFFER_SIZE - DMA_GetCurrDataCounter(DMA1_Channel5); if(rx_len > 0) { // 处理接收到的数据 Process_Complete_Frame(dma_buffer, rx_len); // 重置DMA DMA_Cmd(DMA1_Channel5, DISABLE); DMA_SetCurrDataCounter(DMA1_Channel5, BUFFER_SIZE); DMA_Cmd(DMA1_Channel5, ENABLE); } } }

3.错误处理

不要忘记处理通信错误:

if(USART_GetITStatus(USART1, USART_IT_ORE) != RESET || USART_GetITStatus(USART1, USART_IT_NE) != RESET || USART_GetITStatus(USART1, USART_IT_FE) != RESET) { // 处理溢出、噪声、帧错误 USART_ClearITPendingBit(USART1, USART_IT_ORE | USART_IT_NE | USART_IT_FE); // 重置接收状态 Reset_Rx_State(); }

五、性能优化建议

  1. 使用双缓冲区:一个用于接收,一个用于处理,避免数据竞争

  2. 合理设置超时时间:根据波特率调整,通常为3-5个字符时间

  3. 避免在中断中长时间处理:只做必要的标志设置,数据处理放在主循环

  4. 使用RTOS的消息队列:在中断中发送消息,在任务中处理

总结

判断USART接收数据是否完成有多种方法,选择哪种方法取决于:

  • 数据格式(固定长度/可变长度)

  • 协议要求

  • 系统资源

  • 可靠性要求

对于大多数应用,超时判断法长度字段法的组合是最佳选择。对于高性能要求,DMA+IDLE中断是不二之选。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 13:52:36

C++26契约编程新特性全曝光(post条件实战指南)

第一章&#xff1a;C26契约编程中post条件的核心概念在C26标准的演进中&#xff0c;契约编程&#xff08;Contracts&#xff09;作为提升代码可靠性与可维护性的关键特性&#xff0c;得到了进一步完善。其中&#xff0c;post条件&#xff08;Postconditions&#xff09;用于规定…

作者头像 李华
网站建设 2026/4/23 13:52:20

为什么你的线程池总出问题?:深入剖析资源泄漏的3个致命根源

第一章&#xff1a;为什么你的线程池总出问题&#xff1f;——资源泄漏的宏观视角在高并发系统中&#xff0c;线程池是提升性能的关键组件&#xff0c;但不当使用往往导致资源泄漏&#xff0c;最终引发服务崩溃或响应延迟飙升。资源泄漏并非总是由显式错误引起&#xff0c;更多…

作者头像 李华
网站建设 2026/4/23 12:38:56

vue+uniapp基于微信小程序的老人智能服药提醒的设计与实现

文章目录摘要主要技术与实现手段系统设计与实现的思路系统设计方法java类核心代码部分展示结论源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;摘要 随着老龄化社会的发展&#xff0c;老年人的健康管理问题日益突出&#xff0c;其中服药依…

作者头像 李华
网站建设 2026/4/23 9:59:09

vue+uniapp微信小程序的基于微信小程序电影推荐评分系统视频播放器ylw

文章目录摘要主要技术与实现手段系统设计与实现的思路系统设计方法java类核心代码部分展示结论源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;摘要 该系统基于Vue.js和UniApp框架开发&#xff0c;专为微信小程序平台设计&#xff0c;旨在…

作者头像 李华
网站建设 2026/4/23 12:35:28

你还在手动验证函数返回?C++26 post条件让编译器替你完成

第一章&#xff1a;你还在手动验证函数返回&#xff1f;C26 post条件让编译器替你完成现代C开发中&#xff0c;确保函数行为的正确性是保障系统稳定的关键。以往开发者常依赖断言&#xff08;assert&#xff09;或手动检查来验证函数返回值是否符合预期&#xff0c;这种方式不仅…

作者头像 李华