参考文章:http://blog.csdn.net/linyt/article/details/53355355

本文参考linux系统中 kfifo缓冲区实现.由于没有涉及到锁,在多线程环境下,只适用于 单生产者 + 单消费者 模型.

fifo_buffer.h

#ifndef FIFO_BUFFER_H_
#define FIFO_BUFFER_H_
#include <stdint.h> class FifoBuffer
{
public:
enum
{
enmDefaultBufferLen = ,
};
public:
FifoBuffer(const uint32_t size = enmDefaultBufferLen);
~FifoBuffer();
public:
uint32_t Put(const char *buf, const uint32_t bufLen);
uint32_t Get(char buf[], const uint32_t maxBufLen);
uint32_t Size(){ return size; }
uint32_t EmptySize();
uint32_t UsedSize();
const char *Buffer(){ return buffer; }
private:
uint32_t Min(uint32_t left, uint32_t right){ return left > right ? right : left; }
private:
uint32_t size;
volatile uint32_t in;
volatile uint32_t out;
char *buffer;
};
#endif

fifo_buffer.cpp

#include <malloc.h>
#include <new>
#include <algorithm>
#include "fifo_buffer.h" //////////////////////////////////////////////////////////////////////////
// when out < in
// | |
// |----------------------------------------|
// 0 out|~~~~~~~~~~|in size
//////////////////////////////////////////////////////////////////////////
// when out > in
// | |
// |----------------------------------------|
// 0~~~~~~~|in out|~~~~~~~~~~~~~~~~~~size
////////////////////////////////////////////////////////////////////////// FifoBuffer::FifoBuffer(const uint32_t size /*= enmDefaultBufferLen*/) :in(), out(), size()
{
buffer = new (std::nothrow) char[size];
if (!buffer){ return; }
memset(buffer, , size);
this->size = size;
} FifoBuffer::~FifoBuffer()
{
if (buffer)
{
delete[] buffer;
}
} uint32_t FifoBuffer::Put(const char *buf, const uint32_t bufLen)
{
uint32_t lengthToPut = Min(bufLen, EmptySize());
/* first put the data starting from fifo->in to buffer end */
uint32_t len = Min(lengthToPut, size - (in % size));
memcpy(buffer + (in % size), buf, len);
/* then put the rest (if any) at the beginning of the buffer */
memcpy(buffer, buf + len, lengthToPut - len);
in += lengthToPut;
return lengthToPut;
} uint32_t FifoBuffer::Get(char buf[], const uint32_t maxBufLen)
{
uint32_t lengthToGet = Min(maxBufLen, UsedSize());
/* first get the data from fifo->out until the end of the buffer */
uint32_t len = Min(lengthToGet, size - (out % size));
memcpy(buf, buffer + (out % size), len);
/* then get the rest (if any) from the beginning of the buffer */
memcpy(buf + len, buffer, lengthToGet - len);
out += lengthToGet;
return lengthToGet;
} uint32_t FifoBuffer::EmptySize()
{
return size - in + out;
} uint32_t FifoBuffer::UsedSize()
{
return in - out;
}

测试代码:

#include <stdio.h>
#include <fstream>
#include <windows.h>
#include <thread> #include "fifo_buffer.h"
const char *fileName = "data.txt";
enum
{
enummaxBufLen = ,
}; void WriteToFile(const char *fileName, const char *buf, const uint32_t bufLen);
void ReadFromFile(const char *fileName, char buf[], const uint32_t maxBufLen);
void GenTestFile(); void PutSomeBytes(const char *oriBuf, const uint32_t bufLen, FifoBuffer &fifoBuf);
void GetSomeBytes(FifoBuffer &fifoBuf, char buf[], const uint32_t maxBufLen); int32_t main()
{
GenTestFile();
FifoBuffer fifoBuf(enummaxBufLen + );
char *oriBuf = new char[enummaxBufLen + ];
char *putBuf = new char[enummaxBufLen + ];
memset(oriBuf, , enummaxBufLen + );
memset(putBuf, , enummaxBufLen + );
ReadFromFile(fileName, oriBuf, enummaxBufLen);
//////////////////////////////////////////////////////////////////////////
std::thread put(PutSomeBytes, oriBuf, enummaxBufLen, std::ref(fifoBuf)); std::thread get(GetSomeBytes, std::ref(fifoBuf), putBuf, enummaxBufLen); put.join();
get.join(); printf("%s\n%d\n\n", fifoBuf.Buffer(), strlen(fifoBuf.Buffer())); printf("%s\n%d\n\n", putBuf, strlen(putBuf));
system("pause");
} void WriteToFile(const char *fileName, const char *buf, const uint32_t bufLen)
{
std::ofstream outFile(fileName, std::ios::out);
if (!outFile){ return; }
outFile.write(buf, bufLen);
outFile.close();
} void ReadFromFile(const char *fileName, char buf[], const uint32_t maxBufLen)
{
std::ifstream inFile(fileName, std::ios::in);
if (!inFile){ return; }
inFile.read(buf, maxBufLen);
inFile.close();
} void GenTestFile()
{
char *buf = new char[enummaxBufLen];
for (uint32_t i = ; i < enummaxBufLen; ++i)
{
buf[i] = i % + '';
}
WriteToFile(fileName, buf, enummaxBufLen);
delete[] buf;
} void PutSomeBytes(const char *oriBuf, const uint32_t bufLen, FifoBuffer &fifoBuf)
{
static uint32_t offset = ;
while (offset < bufLen)
{
int32_t byteCount = rand() % ;
offset += fifoBuf.Put(oriBuf + offset, byteCount);
Sleep();
}
} void GetSomeBytes(FifoBuffer &fifoBuf, char buf[], const uint32_t maxBufLen)
{
static uint32_t offset = ;
while (offset < maxBufLen)
{
int32_t byteCount = rand() % ;
offset += fifoBuf.Get(buf + offset, byteCount);
}
}

C++ 环形缓冲区的实现的更多相关文章

  1. linux device driver —— 环形缓冲区的实现

    还是没有接触到怎么控制硬件,但是在书里看到了一个挺巧妙的环形缓冲区实现. 此环形缓冲区实际为一个大小为bufsize的一维数组,有一个rp的读指针,一个wp的写指针. 在数据满时写进程会等待读进程读取 ...

  2. 35.Linux-分析并制作环形缓冲区

    在上章34.Linux-printk分析.使用printk调试驱动里讲述了: printk()会将打印信息存在内核的环形缓冲区log_buf[]里, 可以通过dmesg命令来查看log_buf[] 1 ...

  3. input子系统事件处理层(evdev)的环形缓冲区【转】

    在事件处理层(evdev.c)中结构体evdev_client定义了一个环形缓冲区(circular buffer),其原理是用数组的方式实现了一个先进先出的循环队列(circular queue), ...

  4. 环形缓冲区-模仿linux kfifo【转】

    转自:https://blog.csdn.net/vertor11/article/details/53741681 struct kfifo{ uint8_t *buffer; uint32_t i ...

  5. linux网络编程--Circular Buffer(Ring Buffer) 环形缓冲区的设计与实现【转】

    转自:https://blog.csdn.net/yusiguyuan/article/details/18368095 1. 应用场景 网络编程中有这样一种场景:需要应用程序代码一边从TCP/IP协 ...

  6. linux下C语言实现多线程通信—环形缓冲区,可用于生产者(producer)/消费者(consumer)【转】

    转自:http://blog.chinaunix.net/uid-28458801-id-4262445.html 操作系统:ubuntu10.04 前言:     在嵌入式开发中,只要是带操作系统的 ...

  7. STM32进阶之串口环形缓冲区实现(转载)

    转载自微信公众号“玩转单片机”,感谢原作者“杰杰”. 队列的概念 在此之前,我们来回顾一下队列的基本概念:队列 (Queue):是一种先进先出(First In First Out ,简称 FIFO) ...

  8. 嵌入式框架Zorb Framework搭建二:环形缓冲区的实现

    我是卓波,我是一名嵌入式工程师,我万万没想到我会在这里跟大家吹牛皮. 嵌入式框架Zorb Framework搭建过程 嵌入式框架Zorb Framework搭建一:嵌入式环境搭建.调试输出和建立时间系 ...

  9. 环形缓冲区的应用ringbuffer

    在嵌入式开发中离不开设备通信,而在通信中稳定性最高的莫过于环形缓冲区算法, 当读取速度大于写入速度时,在环形缓冲区的支持下不会丢掉任何一个字节(硬件问题除外). 在通信程序中,经常使用环形缓冲区作为数 ...

  10. 环形缓冲区实现类(Delphi)

    环形缓冲区的用途及原理可以去百度资料狠多的,这里就不介绍了.直接贴代码.代码分别用D7,XE2编译测试 源码下载 http://files.cnblogs.com/lwm8246/uCircleBuf ...

随机推荐

  1. textarea输入回车,前台jsp不回车问题解决

    jsp引入标签: <%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"% ...

  2. NTC 热敏电阻温度计算公式

    Rt = R *EXP(B*(1/T1-1/T2))这里T1和T2指的是K度即开尔文温度,K度=273.15(绝对温度)+摄氏度:其中T2=(273.15+25)Rt 是热敏电阻在T1温度下的阻值: ...

  3. Sql常用语法以及名词解释

    Sql常用语法以及名词解释 SQL分类: DDL—数据定义语言(CREATE,ALTER,DROP,DECLARE) DML—数据操纵语言(SELECT,DELETE,UPDATE,INSERT) D ...

  4. springMVC 的工作原理和机制

    工作原理上面的是springMVC的工作原理图: 1.客户端发出一个http请求给web服务器,web服务器对http请求进行解析,如果匹配DispatcherServlet的请求映射路径(在web. ...

  5. MVC 好记星不如烂笔头之 ---> 全局异常捕获以及ACTION捕获

    public class BaseController : Controller { /// <summary> /// Called after the action method is ...

  6. 简述抽象和封装,对你学习Java有一些作用

    作为一个Java 初学者,对Java的理解可能有些片面,甚至有些错误的理解,对于观看此处的您,希望您选择性观看!!! 天知道我为什么选择学习编程,我不爱编程,但是我既然学习了,还是会努力学习的,在此分 ...

  7. T-SQL 基础学习 02

    数据库设计 定义 数据库设计就是将数据库中的数据实体以及这些数据实体之间关系,进行规划和结构化的过程 在需求分析阶段,设计数据库的一般步骤 A. 收集相信 B. 标识实体 C. 标记每个实体需要存储的 ...

  8. Spring Boot下配置MyBatis多数据源

    http://m.blog.csdn.net/article/details?id=51481911

  9. XVI Open Cup named after E.V. Pankratiev. GP of SPB

    A. Bubbles 枚举两个点,求出垂直平分线与$x$轴的交点,答案=交点数+1. 时间复杂度$O(n^2\log n)$. #include<cstdio> #include<a ...

  10. BZOJ4078 : [Wf2014]Metal Processing Plant

    设$D(A)\leq D(B)$,从小到大枚举$D(A)$,双指针从大到小枚举$D(B)$. 那么对于权值不超过$D(A)$的边,可以忽略. 对于权值介于$(D(A),D(B)]$之间的边,需要满足那 ...