【星云 Orbit-STM32F4】07. 用判断数据尾来接收据的串口通用程序框架
【星云 Orbit-STM32F4】用判断数据尾来接收一串数据的串口通用程序框架
摘要
本文介绍了一种基于STM32F407微控制器的串口数据接收通用程序框架。该框架通过判断数据尾来实现一串数据的完整接收,适用于需要可靠数据传输的应用场景。本文从零开始,详细讲解了STM32F407串口基础知识、配置步骤、HAL库函数详解,并提供了完整的代码示例和注释。目标读者为嵌入式开发小白,内容通俗易懂,适合快速上手。
1. 引言
在嵌入式开发中,串口通信是一种常用的通信方式。本文旨在解决一个常见问题:如何通过STM32F407的串口接收一串数据,并通过判断数据尾来确保数据的完整性。
本文将从零开始,逐步讲解以下内容:
- 基础知识:STM32F407串口的基本概念和工作机制。
- 配置步骤:如何手动配置STM32F407的串口(不使用STM32CubeMX)。
- HAL库函数详解:如何使用STM32 HAL库实现串口接收功能。
- 代码实现:提供完整的代码框架和注释。
- 使用示例:通过实际案例展示如何使用该框架。
2. 基础知识
2.1 STM32F407的串口
STM32F407芯片集成了多个USART(Universal Synchronous Asynchronous Receiver Transmitter)模块,支持同步和异步通信模式。本文将使用USART1模块。
2.2 数据尾判断
在串口通信中,数据通常以帧的形式传输。为了确保数据的完整性,我们需要通过特定的标志(数据尾)来判断一帧数据是否传输完成。例如,可以使用固定的字节序列(如 0xEB 0x00 0x55)作为数据尾。
2.3 数据接收流程
数据接收流程如下:
- 初始化:配置串口参数(波特率、数据位、停止位等)。
- 接收数据:通过中断或轮询方式接收数据。
- 判断数据尾:在接收到数据后,检查是否包含数据尾标志。
- 处理数据:如果检测到数据尾,提取有效数据并进行后续处理。
3. 配置步骤
3.1 时钟配置
在使用串口之前,需要配置时钟系统。以下是关键配置步骤:
- 启用AHB1时钟:确保USART1时钟被启用。
- 配置系统时钟:设置系统时钟频率(本文假设为110.592 MHz)。
// 配置时钟
RCC->AHB1ENR |= RCC_AHB1ENR_GPIOAEN; // 启用GPIOA时钟
RCC->APB2ENR |= RCC_APB2ENR_USART1EN; // 启用USART1时钟
3.2 GPIO配置
配置GPIO引脚用于串口通信。本文使用PA9(TX)和PA10(RX)。
// 配置GPIO
GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.Pin = GPIO_PIN_9 | GPIO_PIN_10;
GPIO_InitStructure.Mode = GPIO_MODE_AF_PP;
GPIO_InitStructure.Pull = GPIO_PULLUP;
GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;
GPIO_InitStructure.Alternate = GPIO_AF7_USART1;HAL_GPIO_Init(GPIOA, &GPIO_InitStructure);
3.3 USART配置
配置USART参数,包括波特率、数据位、停止位和校验位。
// 配置USART
UART_HandleTypeDef huart1;huart1.Instance = USART1;
huart1.Init.BaudRate = 9600;
huart1.Init.WordLength = UART_WORDLENGTH_8B;
huart1.Init.StopBits = UART_STOPBITS_1;
huart1.Init.Parity = UART_PARITY_NONE;
huart1.Init.Mode = UART_MODE_RX | UART_MODE_TX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;HAL_UART_Init(&huart1);
3.4 中断配置
启用USART接收中断。
// 配置中断NVIC_EnableIRQ(USART1_IRQn);
4. HAL库函数详解
4.1 HAL_UART_Init
初始化USART模块。
HAL_StatusTypeDef HAL_UART_Init(UART_HandleTypeDef *huart, UART_InitTypeDef *pInitStruct)
4.2 HAL_UART_Transmit
发送数据。
HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)
4.3 HAL_UART_Receive_IT
启用接收中断。
HAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
4.4 USART1_IRQHandler
USART中断服务函数。
void USART1_IRQHandler(void)
{if (__HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE) != RESET){// 处理接收数据}
}
5. 代码实现
5.1 初始化函数
void SystemClock_Config(void)
{// 配置系统时钟
}void MX_USART1_UART_Init(void)
{// 配置USART1
}void MX_GPIO_Init(void)
{// 配置GPIO
}
5.2 数据接收函数
void USART1_IRQHandler(void)
{static uint8_t rxBuffer[RC_BUFFER_SIZE] = {0};static uint16_t rxIndex = 0;if (__HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE) != RESET){rxBuffer[rxIndex++] = HAL_UART_Read_RX(&huart1);if (rxIndex >= RC_BUFFER_SIZE){// 数据尾判断if (CheckDataTail(rxBuffer, rxIndex)){// 处理数据ProcessData(rxBuffer, rxIndex);rxIndex = 0;}}}
}
5.3 数据处理函数
bool CheckDataTail(uint8_t *data, uint16_t length)
{// 检查数据尾
}void ProcessData(uint8_t *data, uint16_t length)
{// 处理数据
}
6. 使用示例
6.1 初始化
int main(void)
{// 初始化HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_USART1_UART_Init();// 启用接收中断HAL_UART_Receive_IT(&huart1, rxBuffer, RC_BUFFER_SIZE);while (1){// 主循环}
}
6.2 数据接收
void USART1_IRQHandler(void)
{// 中断处理
}
7. 总结
本文提供了一种基于STM32F407的串口数据接收通用程序框架。通过判断数据尾,确保了数据的完整性。本文从零开始,详细讲解了配置步骤和代码实现,适合嵌入式开发小白快速上手。
8. 附录
8.1 完整代码
#include "stm32f4xx_hal.h"#define RC_BUFFER_SIZE 100 // 接收缓冲区大小
#define DATA_TAIL_SIZE 3 // 数据尾长度uint8_t rxBuffer[RC_BUFFER_SIZE] = {0};
uint16_t rxIndex = 0;UART_HandleTypeDef huart1;void SystemClock_Config(void)
{// 配置系统时钟RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLM = 25;RCC_OscInitStruct.PLL.PLLN = 336;RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV4;RCC_OscInitStruct.PLL.PLLQ = 7;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){// 配置错误while (1);}RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK){// 配置错误while (1);}
}void MX_GPIO_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;__HAL_RCC_GPIOA_CLK_ENABLE();GPIO_InitStructure.Pin = GPIO_PIN_9 | GPIO_PIN_10;GPIO_InitStructure.Mode = GPIO_MODE_AF_PP;GPIO_InitStructure.Pull = GPIO_PULLUP;GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;GPIO_InitStructure.Alternate = GPIO_AF7_USART1;HAL_GPIO_Init(GPIOA, &GPIO_InitStructure);
}void MX_USART1_UART_Init(void)
{huart1.Instance = USART1;huart1.Init.BaudRate = 9600;huart1.Init.WordLength = UART_WORDLENGTH_8B;huart1.Init.StopBits = UART_STOPBITS_1;huart1.Init.Parity = UART_PARITY_NONE;huart1.Init.Mode = UART_MODE_RX | UART_MODE_TX;huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart1.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart1) != HAL_OK){// 配置错误while (1);}// 启用接收中断HAL_UART_Receive_IT(&huart1, rxBuffer, RC_BUFFER_SIZE);
}void USART1_IRQHandler(void)
{if (__HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE) != RESET){rxBuffer[rxIndex++] = HAL_UART_Read_RX(&huart1);if (rxIndex >= RC_BUFFER_SIZE){if (CheckDataTail(rxBuffer, rxIndex)){ProcessData(rxBuffer, rxIndex);rxIndex = 0;}}}
}bool CheckDataTail(uint8_t *data, uint16_t length)
{if (length < DATA_TAIL_SIZE){return false;}// 检查数据尾if (data[length - 3] == 0xEB && data[length - 2] == 0x00 && data[length - 1] == 0x55){return true;}return false;
}void ProcessData(uint8_t *data, uint16_t length)
{// 处理数据
}int main(void)
{HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_USART1_UART_Init();while (1){// 主循环}
}
8.2 思维导图
以下是基于STM32F407的串口数据接收程序框架的思维导图:

流程图说明
-
开始:程序启动。
-
系统初始化:配置STM32F407的硬件和外设。
- 配置时钟:配置系统时钟,确保外设时钟正确。
- 配置GPIO:配置与USART相关的GPIO引脚(如TX和RX引脚)。
- 配置USART:初始化USART外设,设置波特率、数据位、停止位和校验位。
-
数据接收:启用USART的接收中断,等待数据接收。
- 启用接收中断:配置USART的中断,使能接收中断。
- 中断服务函数:当接收到数据时,进入中断服务函数,读取接收到的数据。
-
数据处理:对接收到的数据进行处理。
- 检查数据尾:检查数据是否完整,确认数据尾。
- 处理数据:对接收到的数据进行解析或存储。
-
主循环:程序进入主循环,执行轮询任务。
- 执行轮询任务:在主循环中,执行一些周期性任务(如状态监控、按键扫描等)。
-
结束:程序结束。
8.3 状态转换图

- 初始化:程序开始,进入初始化状态。
- 配置时钟:配置系统时钟,确保外设时钟正确。
- 配置GPIO:配置与USART相关的GPIO引脚(如TX和RX引脚)。
- 配置USART:初始化USART外设,设置波特率、数据位、停止位和校验位。
- 启用接收中断:配置USART的中断,使能接收中断。
- 接收数据:等待数据接收。
- 接收中断:当接收到数据时,触发中断服务函数。
- 读取数据:在中断服务函数中,读取接收到的数据。
- 检查数据尾:检查数据尾是否存在,以确定数据是否完整。
- 数据尾存在:如果检测到数据尾,处理数据。
- 处理数据:对接收到的数据进行解析或存储。
- 重置接收索引:在数据处理完成后,重置接收索引,准备接收新的数据。
- 数据尾不存在:如果未检测到数据尾,继续接收数据。
- 主循环:程序进入主循环,执行轮询任务。
- 执行轮询任务:在主循环中,执行一些周期性任务(如状态监控、按键扫描等)。
- 结束:程序结束。
相关文章:
【星云 Orbit-STM32F4】07. 用判断数据尾来接收据的串口通用程序框架
【星云 Orbit-STM32F4】用判断数据尾来接收一串数据的串口通用程序框架 摘要 本文介绍了一种基于STM32F407微控制器的串口数据接收通用程序框架。该框架通过判断数据尾来实现一串数据的完整接收,适用于需要可靠数据传输的应用场景。本文从零开始,详细讲…...
授权与认证之jwt(一)创建Jwt工具类
JWT的Token要经过加密才能返回给客户端,包括客户端上传的Tokn,后端项目需要验证核 实。于是我们需要一个WT工具类,用来加密Token和验证Token的有效性。 一、导入依赖 <dependency><groupId>com.auth0</groupId><artifactId>jav…...
Kubernetes Service服务发现dns之CoreDNS
文章目录 背景什么是Service、服务发现、Endpoint什么是CoreDNSCoreDNS 的工作原理 常用命令coredns 运行状态根据服务名,判断某个服务dns解析是否正常 背景 Kubernetes 集群内部的服务发现是微服务架构的核心基础,而 DNS 服务则是实现这一机制的关键组…...
Spring Boot 测试:单元、集成与契约测试全解析
一、Spring Boot 分层测试策略 Spring Boot 应用采用经典的分层架构,不同层级的功能模块对应不同的测试策略,以确保代码质量和系统稳定性。 Spring Boot 分层架构: Spring Boot分层架构 A[客户端] -->|HTTP 请求| B[Controller 层] …...
用友NC系列漏洞检测利用工具
声明!本文章所有的工具分享仅仅只是供大家学习交流为主,切勿用于非法用途,如有任何触犯法律的行为,均与本人及团队无关!!! 目录标题 YongYouNcTool启动及适配环境核心功能界面预览一键检测命令执…...
PostgreSQL 创建表格
PostgreSQL 创建表格 在数据库管理中,表格(Table)是数据存储的基础。PostgreSQL作为一款强大的开源对象关系型数据库管理系统(ORDBMS),创建表格是其最基本的功能之一。本文将详细讲解如何在PostgreSQL中创…...
一周一个Unity小游戏2D反弹球游戏 - 球的死区及球重生
前言 本文将实现当球弹到球板下方的死亡区域后,球会被重置到球板上发射点,并且重置物理状态的逻辑。 创建球的死亡区 之前创建的在屏幕下方的空气墙碰撞体可以将其Is Trigger勾选上,让其成为一个触发器,用来检测球是否进入该区域,如下。 创建一个脚本名为Deadzone…...
本地部署 DeepSeek:从 Ollama 配置到 Spring Boot 集成
前言 随着人工智能技术的迅猛发展,越来越多的开发者希望在本地环境中部署和调用 AI 模型,以满足特定的业务需求。本文将详细介绍如何在本地环境中使用 Ollama 配置 DeepSeek 模型,并在 IntelliJ IDEA 中创建一个 Spring Boot 项目来调用该模型…...
vue3:三项目增加404页面
一、路由添加 1、官网地址 带参数的动态路由匹配 | Vue Routerhttps://router.vuejs.org/zh/guide/essentials/dynamic-matching.html 2、复制核心语句 { path: /:pathMatch(.*)*, name: NotFound, component: NotFound } 3、粘贴到路由index.js中 4、建立页面 在view文件夹…...
MCAL(Microcontroller Abstraction Layer)介绍
目录 MCAL的核心作用 MCAL的模块组成 1. 微控制器驱动(Microcontroller Drivers) 2. I/O驱动(DIO, PWM, ADC等) 3. 通信驱动(Communication Drivers) 4. 存储驱动(Memory Drivers…...
爬虫:PhantomJS的详细使用和实战案例
文章目录 一、PhantomJS介绍1.1 什么是 PhantomJS1.2 PhantomJS 的特点与优势二、PhantomJS 的安装2.1 在 macOS 上安装 PhantomJS2.2 在 Linux 上安装 PhantomJS2.3 在 Windows 上安装 PhantomJS2.4 验证安装三、PhantomJS 的基本使用3.1 示例 1:打开网页并截图3.2 示例 2:获…...
目标检测——数据处理
1. Mosaic 数据增强 Mosaic 数据增强步骤: (1). 选择四个图像: 从数据集中随机选择四张图像。这四张图像是用来组合成一个新图像的基础。 (2) 确定拼接位置: 设计一个新的画布(输入size的2倍),在指定范围内找出一个随机点(如…...
深度学习工程师的技术图谱和学习路径
在构建一个深度学习工程师的技术图谱时,按照“技能树与能力模型”的结构可以帮助清晰地展示出技术体系的层次化关系,帮助学习者更好地理解每个技术点的依赖与顺序。 深度学习工程师的技术图谱和学习路径 以下是深度学习工程师的技能树,包括从基础到进阶的学习路径,以及对…...
Qt 文件操作+多线程+网络
文章目录 1. 文件操作1.1 API1.2 例子1,简单记事本1.3 例子2,输出文件的属性 2. Qt 多线程2.1 常用API2.2 例子1,自定义定时器 3. 线程安全3.1 互斥锁3.2 条件变量 4. 网络编程4.1 UDP Socket4.2 UDP Server4.3 UDP Client4.4 TCP Socket4.5 …...
如何使用ArcGIS Pro制作横向图例:详细步骤与实践指南
ArcGIS Pro,作为Esri公司推出的新一代地理信息系统(GIS)平台,以其强大的功能和灵活的操作界面,在地理数据处理、地图制作和空间分析等领域发挥着重要作用。 在地图制作过程中,图例作为地图的重要组成部分&…...
Kotlin 嵌套类和内部类
在Kotlin中,嵌套类(Nested Class)和内部类(Inner Class)是两种不同的类,它们在定义和使用上有一些区别。 1.嵌套类(Nested Classes)默认是静态的(即等同于Java中的stati…...
蓝蝶(BlueStacks)模拟器Root、Magisk、LSPosed及Shamiko框架安装与过应用检测指南
蓝蝶(BlueStacks)模拟器Root、Magisk、LSPosed及Shamiko框架安装与过应用检测指南 蓝蝶bluestacks模拟器root和magisk以及Lsposed和shamiko框架的安装过应用检测 一、引言 蓝蝶(BlueStacks)模拟器是一款广受欢迎的安卓模拟器&…...
OpenCV计算摄影学(6)高动态范围成像(HDR imaging)
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 高动态范围成像(HDR imaging)是一种图像处理技术,旨在通过增加图像的动态范围来更准确地表示真实世界的亮度差…...
[ComfyUI][AI生图]如何在Comfyui中安装插件管理器
如何在ComfyUI便携版中安装插件管理器 在现代软件环境中,图形用户界面(GUI)提供了一种直观的方式来与应用程序交互。ComfyUI是一个出色的GUI框架,它使用户能够通过图形化方式配置和管理他们的应用程序。特别是ComfyUI的便携版,它允许用户在没有安装的情况下使用这一工具,…...
初探Ollama与deepseek
什么是Ollama?它与大模型有什么联系? 简单说,Ollama就像是你电脑上的一个 “大模型小助手”。 以前,很多强大的大语言模型,比如能回答各种问题、写文章、翻译等的那些模型,要么只能在网上的服务器上用&am…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
华为云AI开发平台ModelArts
华为云ModelArts:重塑AI开发流程的“智能引擎”与“创新加速器”! 在人工智能浪潮席卷全球的2025年,企业拥抱AI的意愿空前高涨,但技术门槛高、流程复杂、资源投入巨大的现实,却让许多创新构想止步于实验室。数据科学家…...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
C++:std::is_convertible
C++标志库中提供is_convertible,可以测试一种类型是否可以转换为另一只类型: template <class From, class To> struct is_convertible; 使用举例: #include <iostream> #include <string>using namespace std;struct A { }; struct B : A { };int main…...
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...
免费数学几何作图web平台
光锐软件免费数学工具,maths,数学制图,数学作图,几何作图,几何,AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
如何应对敏捷转型中的团队阻力
应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中,明确沟通敏捷转型目的尤为关键,团队成员只有清晰理解转型背后的原因和利益,才能降低对变化的…...
