0

0

C++如何实现简单的环形缓冲区_C++在嵌入式通信中的数据存储【案例】

冰火之心

冰火之心

发布时间:2026-02-06 15:22:02

|

171人浏览过

|

来源于php中文网

原创

环形缓冲区核心是用数组模拟首尾相连队列,靠volatile head/tail指针控制读写,采用“预留空位”法判满((tail+1)&(cap-1)==head),配合位运算与DMB内存屏障实现裸机下无锁安全通信。

c++如何实现简单的环形缓冲区_c++在嵌入式通信中的数据存储【案例】

环形缓冲区的核心实现逻辑是什么

环形缓冲区本质是用数组模拟首尾相连的队列,靠两个指针(headtail)控制读写位置,避免数据搬移。在嵌入式通信中,它常用于串口、CAN 或 SPI 接收中断服务程序(ISR)与主循环之间的解耦——ISR 只管往里写,主循环只管往外读,不加锁也能安全运行(前提是单生产者 + 单消费者,且不跨线程)。

关键不是“怎么封装类”,而是“怎么保证边界不越界、不覆盖、不漏读”。最简实现只需三个成员:buffer(固定大小数组)、head(下一次读的位置)、tail(下一次写的位置),以及一个预设容量 capacity

常见错误现象:head == tail 时无法区分“空”和“满”;直接用 (tail + 1) % capacity 判断是否满,但未预留一个空位;在 ISR 中修改 tail 后没做内存屏障(ARM Cortex-M 上可能被编译器乱序优化)。

  • 推荐用“预留一个空位”法:缓冲区实际可用长度为 capacity - 1full() 条件是 (tail + 1) % capacity == head
  • headtail 必须声明为 volatile(若在 ISR 和主循环间共享),或使用 std::atomic(C++11 起,但嵌入式常禁用 STL)
  • 容量建议设为 2 的幂(如 64、256),方便用位运算替代取模:tail = (tail + 1) & (capacity - 1),省去除法开销

如何在裸机嵌入式环境里安全地用 C++ 实现

裸机(无 OS、无 STL)下不能依赖 std::queuestd::vector,必须手写、零动态分配、所有变量静态或上。典型结构体如下:

立即学习C++免费学习笔记(深入)”;

struct RingBuffer {
    uint8_t buffer[256];
    volatile uint16_t head;
    volatile uint16_t tail;
    static const uint16_t capacity = 256;
};

注意:uint16_t 足够覆盖 64KB 缓冲区,但若容量 ≤ 256,可用 uint8_t 节省空间;volatile 是强制要求,否则编译器可能缓存 head/tail 值导致读写逻辑失效。

写操作(如 UART RX ISR 中):

ThinkAny
ThinkAny

一个RAG驱动的AI搜索引擎,由独立开发者idoubi开发

下载
  • 先判断是否满:if ((tail + 1) & (capacity - 1) != head)(假设 capacity 是 2 的幂)
  • 写入:buffer[tail] = byte;
  • 更新 tail:tail = (tail + 1) & (capacity - 1);
  • 关键:在 tail 更新后加一条 __DMB();(ARM DMB 内存屏障指令),防止写 buffer 和写 tail 被重排

读操作(主循环中):

  • 检查是否空:if (head != tail)
  • 读出:byte = buffer[head];
  • 更新 head:head = (head + 1) & (capacity - 1);
  • 同样需要 __DMB()(读场景通常可省,但对称写更稳妥)

为什么不能直接用 std::array + std::atomic

在部分支持 C++11 的嵌入式工具链(如 ARM GCC 9+)中,std::atomic 理论上可行,但实际踩坑多:

  • 某些 Cortex-M0/M0+ 芯片不支持原子加减硬件指令,std::atomic 会退化为锁实现(需全局互斥量),而裸机根本没锁基础设施
  • std::array 本身没问题,但若搭配 std::atomic 使用,编译器可能生成非紧凑代码,增加 ROM 占用
  • 标准库头文件(如 )可能隐式拉入异常处理或 RTTI 支持,违反嵌入式“零开销抽象”原则
  • 调试困难:GDB 对 std::atomic 的 volatile 行为支持不一,容易误判读写顺序

结论:裸机环境下,显式 volatile + 手动内存屏障比依赖 std::atomic 更可控、更轻量、更容易验证。

中断与主循环共用时最容易忽略的细节

真正出问题的往往不是算法,而是上下文切换的微小疏漏:

  • 缓冲区地址必须位于非 cache 区域(如 STM32 的 SRAM1),或写完后调用 SCB_CleanDCache_by_Addr()(若开启 D-Cache);否则主循环读到的是脏 cache 数据
  • headtail 必须字节对齐(自然满足),但若结构体打包(__attribute__((packed))),要确认编译器没插入填充字节破坏地址连续性
  • UART 中断若频繁触发(如 1Mbps 连续流),写操作必须极简——禁止在 ISR 里调用任何函数(包括 printf)、禁止浮点运算、禁止访问外设寄存器(除非必要)
  • 测试时别只看“能通”,要故意塞满缓冲区再突然停止发送,验证 tail 是否卡死、主循环是否持续读出旧数据

环形缓冲区的健壮性不在代码行数,而在每个内存访问是否被编译器和硬件共同尊重。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

795

2023.08.22

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

75

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

288

2023.11.28

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

282

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

194

2025.07.04

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

69

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

400

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

579

2023.08.10

1688阿里巴巴货源平台入口与批发采购指南
1688阿里巴巴货源平台入口与批发采购指南

本专题整理了1688阿里巴巴批发进货平台的最新入口地址与在线采购指南,帮助用户快速找到官方网站入口,了解如何进行批发采购、货源选择以及厂家直销等功能,提升采购效率与平台使用体验。

49

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 5.2万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号