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

简单理解:单个环形缓冲区 vs 双缓冲区 对比表

对比项单个大环形缓冲区双缓冲区(双缓冲)
解决的核心问题数据不会溢出、不会满保证读到完整一整包、不被打断
读写方式一边写、一边读,同时进行写 A 时读 B,写 B 时读 A,互不干扰
数据完整性可能读到一半旧一半新(脏数据)永远是完整一包,安全干净
会不会满缓冲区够大就不会满也不会满,但重点不是防满
实现难度简单,小白首选稍复杂,用于特定场景
典型用途串口收发、按键、小数据、调试网络帧、音频、屏幕显示、DMA 整包处理
是否需要切换不用切换,一个到底需要来回切换 A/B
你现在是否需要非常需要,必须掌握❌ 暂时不用,进阶才学

一句话背下来

  • 单个大环 = 防满、简单、通用
  • 双缓冲 = 保整包、不被打扰、专用

【方案 1】单个大环形缓冲区(不会满、通用、最简单)

👉 适用:串口、不断收发、一边写一边读、不会丢数据

// 包含固定类型支持,比如 uint8_t = 1个字节 #include <stdint.h> // 包含 true / false #include <stdbool.h> //============================= // 缓冲区大小 // 设大一点 = 永远不会满! //============================= #define RING_BUFFER_SIZE 256 //============================= // 环形缓冲区结构体 // 一个结构体 = 一个完整缓冲区 //============================= typedef struct { // 真正用来存数据的数组 uint8_t buffer[RING_BUFFER_SIZE]; // 写指针:下一个数据要写到哪里 volatile uint16_t head; // 读指针:下一个数据从哪里读 volatile uint16_t tail; } RingBuffer; //============================= // 初始化缓冲区 // 作用:清空数据,指针归零 //============================= void RingBuffer_Init(RingBuffer *rb) { // 写指针回到 0 rb->head = 0; // 读指针回到 0 rb->tail = 0; } //============================= // 写入 1 个字节 // 返回:true=成功,false=满 //============================= bool RingBuffer_Put(RingBuffer *rb, uint8_t data) { // 计算下一个写位置 uint16_t next_head = (rb->head + 1) % RING_BUFFER_SIZE; // 如果下一个写位置 == 读位置 → 满了 if (next_head == rb->tail) { return false; } // 把数据写入当前 head 位置 rb->buffer[rb->head] = data; // 写指针向后移动 rb->head = next_head; return true; } //============================= // 读取 1 个字节 // 返回:true=成功,false=空 //============================= bool RingBuffer_Get(RingBuffer *rb, uint8_t *out_data) { // 如果 head == tail → 没有数据 if (rb->head == rb->tail) { return false; } // 从 tail 位置读出数据 *out_data = rb->buffer[rb->tail]; // 读指针向后移动 rb->tail = (rb->tail + 1) % RING_BUFFER_SIZE; return true; } //============================= // 判断是否为空 //============================= bool RingBuffer_IsEmpty(RingBuffer *rb) { return (rb->head == rb->tail); } //============================= // 判断是否已满 //============================= bool RingBuffer_IsFull(RingBuffer *rb) { uint16_t next_head = (rb->head + 1) % RING_BUFFER_SIZE; return (next_head == rb->tail); }

【方案 2】双缓冲区(保证数据完整、不被打扰)

👉 适用:整包数据、网络、音频、DMA、不能边读边改

// 固定头文件 #include <stdint.h> #include <stdbool.h> //============================= // 每个缓冲区大小 //============================= #define DOUBLE_BUF_SIZE 128 //============================= // 双缓冲区结构体 //============================= typedef struct { // 两个独立缓冲区 A 和 B uint8_t bufA[DOUBLE_BUF_SIZE]; uint8_t bufB[DOUBLE_BUF_SIZE]; // 当前正在写入的缓冲区(指向 A 或 B) uint8_t *write_buf; // 当前写了多少字节 uint16_t write_pos; // 是否有一整块数据准备好了可以读 bool data_ready; } DoubleBuffer; //============================= // 初始化双缓冲区 //============================= void DoubleBuffer_Init(DoubleBuffer *db) { // 一开始用 A 来写 db->write_buf = db->bufA; // 写位置从0开始 db->write_pos = 0; // 一开始没有数据可读 db->data_ready = false; } //============================= // 写入 1 字节 //============================= bool DoubleBuffer_Write(DoubleBuffer *db, uint8_t data) { // 如果有数据还没读,不能写(防止覆盖) if (db->data_ready) { return false; } // 把数据写入当前写缓冲区 db->write_buf[db->write_pos] = data; // 位置往后移 db->write_pos++; // 如果写满了 if (db->write_pos >= DOUBLE_BUF_SIZE) { // 切换缓冲区 if (db->write_buf == db->bufA) { db->write_buf = db->bufB; } else { db->write_buf = db->bufA; } // 重置写位置 db->write_pos = 0; // 标记:有一整块数据可以读 db->data_ready = true; } return true; } //============================= // 读取一整块数据 //============================= bool DoubleBuffer_Read(DoubleBuffer *db, uint8_t **out_buf, uint16_t *out_len) { // 没有准备好的数据 if (!db->data_ready) { return false; } // 找到上一个写满的缓冲区 if (db->write_buf == db->bufA) { *out_buf = db->bufB; } else { *out_buf = db->bufA; } // 长度是满包长度 *out_len = DOUBLE_BUF_SIZE; // 清除标志 db->data_ready = false; return true; } //============================= // 是否有完整数据可以读 //============================= bool DoubleBuffer_IsReady(DoubleBuffer *db) { return db->data_ready; }

我再给你最清晰总结(小白必看)

单个环形缓冲区

  • 一个数组
  • 一边写、一边读
  • 不会满
  • 简单、通用
  • 你日常 99% 用这个

双缓冲区

  • 两个独立数组
  • 写满一个再换另一个
  • 读的时候绝对不会被修改
  • 用于整包数据
http://www.jsqmd.com/news/651400/

相关文章:

  • 快速搭建企业级Spring Boot OAuth2认证系统的终极指南
  • 别再复制粘贴了!STM32F103C8T6驱动ADXL345的完整避坑指南(附工程源码)
  • 避坑指南:PetaLinux下AXI Uartlite串口收数据不连续?我的硬件协同调试复盘
  • Python 上下文管理器:原理与应用
  • 别再死记硬背了!一张图搞定华为数通里的网络类型与拓扑(附实战场景联想)
  • 前端微前端进阶:从架构到实践
  • 西门子恒压供水系统程序:详细注释与图纸,一拖多泵组合,水箱无负压模式切换,画面随选更新,PLC...
  • Apollo 10.0 在Ubuntu22.04下的完整环境配置指南
  • 前端PDF预览避坑指南:从Blob转换到vue-pdf分页控制的那些事儿
  • 从X-AnyLabeling到YOLO:一站式JSON标签转换实战指南(附Python脚本)
  • 从模型检测实战看三大逻辑:CTL、PLTL与mu-演算的选型指南
  • 批处理脚本进阶:环境隔离、参数轮转与流式处理
  • 某手App反爬核心sig3算法解析:从Unidbg服务部署到接口调用的完整链路
  • Unity3d Cinemachine篇(一)— 初探Virtual Camera:从零搭建你的首个智能镜头
  • 手把手教你用Glean搭建企业知识图谱:从Slack到Confluence的完整配置流程
  • 避坑指南:部署完kube-prometheus后,为什么Grafana/Prometheus页面还是打不开?
  • 合宙ESP32C3实战:MPU6500六轴传感器数据读取与校准全解析
  • 用CY7C68013A模拟MDIO时序?这些GPIO配置细节你可能不知道
  • 央视曝光 AI 涉灰产业链:技术红利正被滥用,监管必须跟上
  • 从源码到一键安装包:教你用PyInstaller打包定制版LabelImg(解决闪退和预置标签问题)
  • 《TRAE从入门到精通全攻略》,零基础也能快速上手,助力你快速成长为程序员
  • 雷达信号分析入门:脉内脉间调制到底在玩什么花样?
  • 基于 MATLAB 实现的可视密码图示法设计
  • PCB设计老鸟的AD21 DRC设置清单:如何为你的高速板与低速板定制专属检查规则
  • 终极Windows ISO补丁集成指南:一键制作最新补丁安装镜像的完整教程
  • 科学化学工管理:让教育更高效,让学生更满意
  • DRV8701E双电机驱动电路实战:从原理图困惑到PCB布局的避坑指南
  • Nginx正向代理实战:从源码编译到HTTPS支持的全流程指南
  • 如何用Python自动化脚本破解大麦网抢票难题:技术原理与实战指南
  • 提前72小时预警,巡检提效60%!华电集团联合吉泰智能斩获《火电燃料技术创新大奖》