参考文章: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);
}
}

最新文章

  1. ASP.NET MVC TagBuilder使用
  2. [转]Windows系统中监控文件复制操作的几种方式
  3. Python 时间整理
  4. Ehcache(04)——设置缓存的大小
  5. eclipse 不能切换输入法
  6. OTG线与普通USB线的区别
  7. Java笔记--java一行一行写入或读取数据
  8. CSS display:table属性用法- 轻松实现了三栏等高布局
  9. yum lamp for Centos6.4
  10. linux环境变量入门
  11. 彻底搞明白find命令的-mtime参数的含义【转载】
  12. Microsoft Web Test Recorder在录制时没有显示
  13. unity3D:游戏分解之角色移动和相机跟随
  14. HDU 1864 最大报销额(DP)
  15. linux-0.11 内核源码学习笔记一(嵌入式汇编语法及使用)
  16. spring boot之hello
  17. word产品密钥激活
  18. SQL Server还原数据库
  19. less语法(一)变量与extend
  20. poj3414 Pots(BFS)

热门文章

  1. css中一些常用技巧
  2. Redirect url 路径简单介绍
  3. 简单粗暴将sqlserver表以及数据迁移到oracle
  4. java 注解 + 自定义注解的使用
  5. VNC安装与使用
  6. 参数table_open_cache
  7. python base64的加密与解密
  8. HTML5离线Web应用实战:五步创建成功
  9. 《DSP using MATLAB》示例Example5.7
  10. BZOJ 2115 [Wc2011] Xor ——线性基