news 2026/4/16 12:14:10

UART串口不定长数据接收方法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
UART串口不定长数据接收方法

一、基本概念与问题

在嵌入式系统中,串口(UART)通信时,数据通常以不定长的“帧”为单位发送。串口硬件本身只能识别单个字节的接收完成,无法自动判断一帧数据何时开始和结束。因此,需要通过软件方法来解决帧边界识别问题。

所有方法都基于一个基本的数据管理结构:

#define MAX_BUF_SIZE 200 typedef struct { uint8_t buffer[MAX_BUF_SIZE]; // 数据存储区 uint16_t count; // 已接收字节数 uint16_t length; // 帧长度 uint8_t complete_flag; // 帧完成标志 } UartRxManager; UartRxManager uart1_rx;

二、四种基础实现方法

1. 空闲中断检测法

原理

利用串口硬件的空闲检测功能:当RX引脚在一个字节传输时间后保持高电平,硬件会触发空闲中断,标志着一帧数据传输结束。

实现要点

初始化配置

// 开启空闲中断 __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE); // 启动DMA接收 HAL_UART_Receive_DMA(&huart1, uart1_rx.buffer, MAX_BUF_SIZE);

中断处理

void handle_idle_interrupt(UART_HandleTypeDef *huart) { if(__HAL_UART_GET_FLAG(huart, UART_FLAG_IDLE)) { __HAL_UART_CLEAR_IDLEFLAG(huart); // 停止DMA并计算接收长度 HAL_UART_DMAStop(huart); uart1_rx.length = MAX_BUF_SIZE - __HAL_DMA_GET_COUNTER(huart->hdmarx); uart1_rx.complete_flag = 1; // 重新启动接收 HAL_UART_Receive_DMA(huart, uart1_rx.buffer, MAX_BUF_SIZE); } }

2. 协议解析法

原理

在通信协议中定义固定的帧结构,通过识别帧头、帧长等信息来确定帧边界。

协议示例
字节位置内容说明
00x5A帧头1
10xA5帧头2
2N数据长度
3~N+2数据有效载荷
实现代码
typedef enum { WAIT_HEADER1, WAIT_HEADER2, WAIT_LENGTH, RECEIVING_DATA } RxState; void process_received_byte(uint8_t byte) { static RxState state = WAIT_HEADER1; static uint8_t expected_len = 0; switch(state) { case WAIT_HEADER1: if(byte == 0x5A) { uart1_rx.count = 0; uart1_rx.buffer[uart1_rx.count++] = byte; state = WAIT_HEADER2; } break; case WAIT_HEADER2: if(byte == 0xA5) { uart1_rx.buffer[uart1_rx.count++] = byte; state = WAIT_LENGTH; } else { state = WAIT_HEADER1; // 重新同步 } break; case WAIT_LENGTH: expected_len = byte; uart1_rx.buffer[uart1_rx.count++] = byte; state = RECEIVING_DATA; break; case RECEIVING_DATA: uart1_rx.buffer[uart1_rx.count++] = byte; if(uart1_rx.count >= (expected_len + 3)) { uart1_rx.complete_flag = 1; uart1_rx.length = uart1_rx.count; state = WAIT_HEADER1; } break; } }

3. 超时判断法

原理

基于数据连续性假设:如果在一定时间内没有收到新数据,则认为当前帧已结束。

时间计算

以9600波特率为例:

  • 1个字节传输时间 ≈ 1.04ms (10位/字节 ÷ 9600位/秒)

  • 超时时间建议:1.5-2倍字节时间 ≈ 2ms

实现代码
volatile uint32_t last_receive_time = 0; #define TIMEOUT_MS 2 // 接收中断中调用 void on_byte_received(uint8_t byte) { uart1_rx.buffer[uart1_rx.count++] = byte; last_receive_time = get_current_time(); // 更新时间戳 } // 主循环中检查 void check_timeout(void) { uint32_t current_time = get_current_time(); if(uart1_rx.count > 0 && (current_time - last_receive_time > TIMEOUT_MS)) { uart1_rx.complete_flag = 1; uart1_rx.length = uart1_rx.count; uart1_rx.count = 0; // 准备接收下一帧 } }

4. 环形缓冲区法

原理

中断只负责将数据存入缓冲区,主程序从缓冲区读取并解析数据,实现接收与处理的解耦。

数据结构
typedef struct { uint8_t *data; uint16_t size; uint16_t head; // 写入位置 uint16_t tail; // 读取位置 uint16_t count; // 数据数量 } RingBuffer; void rb_init(RingBuffer *rb, uint8_t *buf, uint16_t size) { rb->data = buf; rb->size = size; rb->head = rb->tail = rb->count = 0; } uint8_t rb_write(RingBuffer *rb, uint8_t byte) { if(rb->count >= rb->size) return 0; rb->data[rb->head] = byte; rb->head = (rb->head + 1) % rb->size; rb->count++; return 1; } uint8_t rb_read(RingBuffer *rb, uint8_t *byte) { if(rb->count == 0) return 0; *byte = rb->data[rb->tail]; rb->tail = (rb->tail + 1) % rb->size; rb->count--; return 1; }
使用方式
// 中断服务程序(极简) void USART1_IRQHandler(void) { uint8_t byte = USART1->DR; rb_write(&rx_buffer, byte); } // 主程序处理 int main(void) { uint8_t byte; while(1) { if(rb_read(&rx_buffer, &byte)) { // 解析协议或处理数据 process_received_byte(byte); } } }

三、方法对比与选择

方法优点缺点适用场景
空闲中断硬件支持、效率高需要硬件支持高速连续数据
协议解析可靠性高、有校验实现较复杂有固定协议
超时判断实现简单精度依赖定时低速应用
环形缓冲区解耦接收与处理内存占用较大多任务系统

四、选择建议

  1. 简单应用:从超时判断法开始,最容易理解和实现

  2. 可靠通信:选择协议解析法,具备错误检测能力

  3. 高效接收:使用空闲中断+DMA,减少CPU干预

  4. 复杂系统:采用环形缓冲区,便于扩展和维护

实际应用中常组合使用,如:空闲中断+协议解析,或环形缓冲区+超时判断。

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

轻松搞定trader-x合约量化策略的实现与优化

我做量化交易已经有一段时间了,经历过不少起起伏伏的过程,像很多人一样,也曾在各种平台上尝试过不同的策略和工具。XTrader是我这几年使用最多的一个平台,它的稳定性和功能还是让我挺满意的,尤其是在执行trader-x合约量…

作者头像 李华
网站建设 2026/4/15 13:43:01

非线性参数的精英学习灰狼优化算法(Matlab)

非线性参数的精英学习灰狼优化算法-- Matlab 改进策略: 1、精英反向学习初始化种群 2、调整收敛因子a 3、改造位置更新公式 仅包含代码,不含讲解 在优化算法的领域中,灰狼优化算法(GWO)以其独特的模拟狼群狩猎行为而受…

作者头像 李华
网站建设 2026/4/15 13:36:37

T600化学镀锡药水国产化:基于工程与材料科学原理的解决方案

摘要 化学镀锡技术在PCB表面处理行业占据重要地位。T600化学镀锡作为一种国产解决方案,旨在解决长期以来高端电镀添加剂依赖进口所带来的成本高、交付延迟以及传统工艺存在的质量缺陷等问题。其主要通过甲基磺酸、硫酸体系的化学镀锡药水,配合配套的去离…

作者头像 李华
网站建设 2026/4/8 19:30:57

网络安全学习路线(全网独家),从入门到实战

很多小伙伴在网上搜索网络安全时,会出来网络安全工程师这样一个职位,它的范围很广,只要是与网络安全挂钩的技术人员都算网络安全工程师,一些小伙伴就有疑问了,网络安全现在真的很火吗? 那么寒哥就带大家看…

作者头像 李华
网站建设 2026/4/8 9:51:25

燃烧室设计学习DAY5:预混燃烧VS扩散燃烧:效率提升之谜

目录 预混燃烧与非预混燃烧的深度解析:机理差异与效率探讨 摘要 第一章 绪论 1.1 燃烧科学背景 1.2 问题的提出 第二章 燃烧的基本物理化学基础 2.1 Damkhler数与燃烧模式 2.2 层流与湍流的影响 第三章 非预混燃烧(扩散燃烧)详解 3…

作者头像 李华