当前位置: 首页 > news >正文

STM32 串口中断接收方式笔记:HAL_UART_Receive_IT vs __HAL_UART_ENABLE_IT

📘 STM32 串口中断接收方式笔记:HAL_UART_Receive_IT vs __HAL_UART_ENABLE_IT


🧠 1. 两者作用简述

函数/宏作用是否配置HAL状态是否调用Rx回调
HAL_UART_Receive_IT()启动一次基于中断的串口接收任务✅ 是✅ 是
__HAL_UART_ENABLE_IT()手动使能某个串口中断(如RXNE)❌ 否❌ 否(除非配合HAL函数)

🔧 2. HAL_UART_Receive_IT() 的工作机制

HAL_UART_Receive_IT(&huart1, rx_buf, 10);
  • 作用:启动一个“异步接收任务”,要求 HAL 接收 10 个字节 存入 rx_buf 中。
  • 内部做了以下几件事:
    1. 设置接收缓冲区指针 huart->pRxBuffPtr
    2. 设置接收长度 huart->RxXferSize
    3. 状态切换为 BUSY_RX
    4. 使能 RXNE 中断
  • 当 HAL 中断服务函数 HAL_UART_IRQHandler() 检测到接收到的数据后,会自动读取数据放入缓冲区,直到接收满为止。
  • 满足接收条件后,会自动调用:
    void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart);
    

注意:接收完成后,不会自动重新启动接收,需要你手动再次调用 HAL_UART_Receive_IT()


⚠️ 3. 使用 __HAL_UART_ENABLE_IT() 的注意事项

__HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);
  • 只是打开了 USART1 的接收中断 RXNE
  • 并不会设置 HAL 的接收状态、缓冲区等内部变量。
  • 如果你还调用了 HAL_UART_IRQHandler(),会因 HAL 状态未准备好,导致:
    • 中断进来了
    • RXNE 置位
    • HAL 不消费数据
    • RXNE 不会被清除
    • 中断一直触发 → 卡死!

✅ 正确用法对比

✅ 【方式1】HAL自动管理方式(推荐新手/结构清晰)
// 初始化接收任务
HAL_UART_Receive_IT(&huart1, rx_buf, 1);// 实现接收完成回调
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{if (huart->Instance == USART1) {// 处理数据 rx_buf[0]HAL_UART_Receive_IT(&huart1, rx_buf, 1);  // 继续接收}
}
✅ 【方式2】裸中断方式(适合自定义接收流程)
// 手动使能接收中断
__HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);void USART1_IRQHandler(void)
{if (__HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE)) {uint8_t ch = (uint8_t)(huart1.Instance->RDR);// 自定义缓冲区存储、协议解析等}
}

🔄 4. 总结对比表

项目HAL_UART_Receive_IT()__HAL_UART_ENABLE_IT() + 手动处理
接收管理HAL帮你管理接收缓冲你自己处理
接收长度固定长度可灵活按字节
回调机制自动调用 RxCpltCallback你自己写中断函数
中断使能内部自动配置你自己配置
出错风险大(忘记清标志位或缓冲区会卡死)
场景推荐简单数据帧、串口透传协议解析、环形缓冲、实时性要求高

🧪 5. 常见问题

问题原因
程序卡在中断里出不来只使能了 RXNE 中断,没有初始化接收任务,RXNE 一直为 1
接收不到数据没有调用 HAL_UART_Receive_IT(),或中断未使能
回调函数不触发没有使用 HAL 接收函数,或 HAL 状态机异常
中断乱跳、多次进中断收数据没及时读 RDR,或 RXNE 没清除
http://www.dtcms.com/a/138229.html

相关文章:

  • 用GitHub Actions实现CI/CD
  • 【uniapp】uni.setClipboardData 方法失效 bug 解决方案
  • 阿里云OSS应对DDoS攻击策略
  • TDengine 存储引擎剖析:数据文件与索引设计(二)
  • 告别昂贵语音合成服务!用GPT-SoVITS生成你的个性化AI语音
  • 【eNSP实验】OSPF单区域配置
  • 高效培训,借助课程编辑器塑造卓越团队​
  • BH1750光照传感器---附代码
  • 2025年4月16日华为笔试第二题200分
  • js逆向分享
  • NAT、代理服务、内网穿透
  • AI 推理与训练优化的核心理论体系建构及关键技术分析框架
  • Leetcode - 周赛445
  • ServletRequestAttributeListener 的用法笔记250417
  • usb重定向qemu前端处理
  • Java排序算法百科全书:原理、实现与实战指南
  • 如何新建一个空分支(不继承 master 或任何提交)
  • SQL刷题记录贴
  • Nodemon vs. PM2:开发与生产环境的 Node.js 部署最佳实践
  • 【AI插件开发】Notepad++ AI插件开发实践:实现对话窗口功能
  • 多 Agent 协作怎么整:从谷歌A2A到多Agent交互方案实现
  • 【hive】Hive对数据库,对表的操作(一)
  • 第六节:React Hooks进阶篇-自定义Hook设计
  • 大模型时代下全场景数据消费平台的智能BI—Quick BI深度解析
  • 【数字图像处理】图像增强
  • King3399(ubuntu文件系统)GDB/GDBServer调试配置
  • 《Cangjie Magic实战手记:用Agent DSL与MCP协议重构智能物流调度系统》——一次从技术困惑到行业落地的探索之旅
  • 当 AI 有了 “万能插头” 和 “通用语言”:MCP 与 A2A 如何重构智能体生态
  • 【JAVA】在idea新加artifact时,点击Build-Build Artifacts时,新加的artifact不能选中
  • Java NIO Java 虚拟线程(微线程)与 Go 协程的运行原理不同 为何Go 能在低配机器上承接10万 Websocket 协议连接