news 2026/4/16 18:20:58

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
简单理解:单个环形缓冲区 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% 用这个

双缓冲区

  • 两个独立数组
  • 写满一个再换另一个
  • 读的时候绝对不会被修改
  • 用于整包数据
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 18:19:50

Fortigate CLI实战:从零到精通的防火墙运维指南

1. Fortigate防火墙CLI入门&#xff1a;从零开始连接设备 第一次接触Fortigate防火墙的命令行界面&#xff08;CLI&#xff09;时&#xff0c;很多新手会感到无从下手。其实只要掌握几个基础步骤&#xff0c;就能快速上手。我刚开始接触Fortigate时也踩过不少坑&#xff0c;现在…

作者头像 李华
网站建设 2026/4/16 18:19:44

Spring Boot 中的事务管理:确保数据一致性

在开发基于Spring Boot的应用程序时,数据一致性和事务管理是至关重要的。尤其是在处理多个表的数据插入时,确保原子性(即所有操作要么全部成功,要么全部失败)是每个开发者都应关注的重点。本文将通过一个具体的实例,探讨如何在Spring Boot中正确地使用事务管理来保证数据…

作者头像 李华