参考文章: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. 转:asp.net TreeView CheckChanged 事件浅谈

    http://blog.csdn.net/xiage/article/details/5128755 在开发中经常可以碰到类似的问题: 想通过一个树父节点的TreeNodeCheckChanged 事 ...

  2. springMVC 学习(一)

    本文主要介绍springmvc的框架原理,并通过一个入门程序展示环境搭建,配置以及部署调试. springmvc是spring框架的一个模块,springmvc和spring无需通过中间整合层进行整合 ...

  3. Mysql Error: Host ‘xxx.xxx.xxx.xxx’ is not allowed to connect to

    1.打开cmd控制台 2. mysql -u root -p 输入密码 3. mysql>grant all privileges on *.* to 'root' @'%' identifie ...

  4. Power BI for Office 365(一)移动端应用

    此篇来自于微软商业智能网站的官方博客团队发布的Power BI在线资料其中的一部分,完整版地址: http://office.microsoft.com/en-us/office365-sharepo ...

  5. Linux下安装JDK多种方式

    一.环境说明: 操作系统:Linux xx- -.el6.x86_64 # SMP Thu Jul :: UTC x86_64 x86_64 x86_64 GNU/Linux jdk版本:java-- ...

  6. MySQL 相关

    Innodb引擎 Innodb引擎提供了对数据库ACID事务的支持,并且实现了SQL标准的四种隔离级别.该引擎还提供了行级锁和外键约束,它的设计目标是处理大容量数据库系统. 但是该引擎不支持FULLT ...

  7. java基础之——DecimalFormat格式化数字

    DecimalFormat可以用来格式化数字,例如用来设定保留多少位小数.设定数字分隔符等. 说方法之前,先介绍一下其常用的几个模式占位符: 0 一个数字# 一个数字,不包括 0. 小数的分隔符的占位 ...

  8. java线程池ThreadPoolExecutor使用简介

    一.简介线程池类为 java.util.concurrent.ThreadPoolExecutor,常用构造方法为:ThreadPoolExecutor(int corePoolSize, int m ...

  9. shell example01

    条件判断 if [[ -e ${1} ]]; then echo "$(tput setaf 2) found ${1} $(tput sgr0)" cat ${1} else e ...

  10. 读取EXCEL数据到内存DataTable

    protected void Page_Load(object sender, EventArgs e) { string filepath = Server.MapPath("~/file ...