写C#的同志一定觉得Byte []比C++的 BYTE * 加 Length的方式好的多。一来,只需要一个对象就可以是表示一段字节流,另一方面,由于C#的特性,不需要象C++那样还要记得删除指针。由于我工作中,需要频繁地试用C#和C++,所以写了个C++的类,以便方便地管理字节流。

很简单,先定义一个类:CMemoryBuffer。字节流内部可以用std::vector<BYTE>来保存,当然,考虑到效率,有些地方处理还是要考虑下。先把代码贴出来,然后解释为什么这么做。

头文件:

#include <vector>
#include <queue>
#include <afxmt.h> using namespace std; class CMemoryBuffer
{
public:
CMemoryBuffer(void);
CMemoryBuffer(const CMemoryBuffer &other);
CMemoryBuffer(const BYTE *tpBytes ,int tiLength);
virtual ~CMemoryBuffer(void); //得到内部指针——const
const BYTE * c_Bytes() const;
//从内部拷贝数据到数据中
BYTE * CopyOut(int &tiLength) const;
//确保tppBytes指向的数据足够大
void CopyTo(const BYTE ** tppBytes, int &tiLenth) const;
//从外部数据拷贝数据
void CopyFrom(const BYTE * tpBytes , int tiLength);
//从外部数据拷贝数据,添加
void Append(const BYTE * tpBytes , int tiLength);
void Append(const BYTE & tByte);
//从外部数据拷贝数据,插入
void Insert(int tiStartIndex,const BYTE * tpBytes , int tiLength); CMemoryBuffer & operator = (const CMemoryBuffer &other); CMemoryBuffer & operator += (const CMemoryBuffer &other); const std::vector<BYTE> &GetBuffer() const { return m_vctBuffer; } void Clear() ; int GetLength() const { return (int)m_vctBuffer.size(); } bool IsEmpty() const { return m_vctBuffer.size() == ; } public:
vector<BYTE> m_vctBuffer;
};

CPP文件:

#include "StdAfx.h"
#include "MemoryBuffer.h" CMemoryBuffer::CMemoryBuffer(void)
{
} CMemoryBuffer::~CMemoryBuffer(void)
{
this->Clear();
} CMemoryBuffer::CMemoryBuffer(const CMemoryBuffer &other)
{
*this = other;
} CMemoryBuffer::CMemoryBuffer(const BYTE *tpBytes ,int tiLength)
{
this->CopyFrom(tpBytes,tiLength);
} void CMemoryBuffer::Clear()
{
vector<BYTE>().swap(this->m_vctBuffer);
} const BYTE * CMemoryBuffer::c_Bytes() const
{
if(this->IsEmpty()) return NULL;
return &m_vctBuffer[];
} BYTE * CMemoryBuffer::CopyOut(int &tiLength) const
{
tiLength = this->GetLength();
if(this->IsEmpty()) return NULL;
BYTE *pBytes = new BYTE[tiLength];
memcpy(pBytes,&m_vctBuffer[],tiLength);
return pBytes;
} void CMemoryBuffer::CopyFrom(const BYTE * tpBytes , int tiLength)
{
this->Clear();
if(tpBytes == NULL || tiLength == ) return;
m_vctBuffer.resize(tiLength,);
memcpy(&m_vctBuffer[],tpBytes,tiLength); } void CMemoryBuffer::Append(const BYTE * tpBytes , int tiLength)
{
if(tpBytes == NULL || tiLength == ) return;
m_vctBuffer.resize(this->GetLength() + tiLength,);
memcpy(&m_vctBuffer[] + this->GetLength() - tiLength,tpBytes,tiLength);
} void CMemoryBuffer::Append(const BYTE & tByte)
{
m_vctBuffer.push_back(tByte); } void CMemoryBuffer::Insert(int tiStartIndex,const BYTE * tpBytes , int tiLength)
{
if(tpBytes == NULL || tiLength == ) return;
int iBufferSize = this->GetLength();
if(tiStartIndex > iBufferSize) return;
if(tiStartIndex == iBufferSize)
{
this->Append(tpBytes,tiLength);
}
else if((tiStartIndex + tiLength) < iBufferSize)
{
memcpy(&m_vctBuffer[] + tiStartIndex,tpBytes,tiLength);
}
else
{
m_vctBuffer.resize(tiStartIndex + tiLength ,);
memcpy(&m_vctBuffer[] + tiStartIndex,tpBytes,tiLength);
} } void CMemoryBuffer::CopyTo(const BYTE ** tppBytes, int &tiLength)const
{
if(tppBytes == NULL || *tppBytes == NULL || this->IsEmpty()) return;
tiLength = this->GetLength();
memcpy(tppBytes,&m_vctBuffer[],tiLength); } CMemoryBuffer & CMemoryBuffer::operator = (const CMemoryBuffer &other)
{
this->Clear();
if (!other.IsEmpty())
{
m_vctBuffer.insert(m_vctBuffer.begin(),other.GetBuffer().begin(),other.GetBuffer().end());
}
return *this;
} CMemoryBuffer & CMemoryBuffer::operator += (const CMemoryBuffer &other)
{
if (!other.IsEmpty())
{
m_vctBuffer.insert(m_vctBuffer.end(),other.GetBuffer().begin(),other.GetBuffer().end());
}
return *this;
}

解释下几点:

1、void CMemoryBuffer::Clear()
    {
     vector<BYTE>().swap(this->m_vctBuffer); 
    }

这地方之所以要这么写,是因为vector有个毛病,clear后内存空间还不释放,需要对象释放后才释放,如果频繁操作一个大的字节流,怕影响   内存性能.

2、void CMemoryBuffer::CopyFrom(const BYTE * tpBytes , int tiLength)
{
 this->Clear();
 if(tpBytes == NULL || tiLength == 0) return;
 m_vctBuffer.resize(tiLength,0);
 memcpy(&m_vctBuffer[0],tpBytes,tiLength);

}

很多人可能会写成一个循环语句:

for(int i = 0; i < tiLength; i ++)

{

m_vctBuffer.push_back(tpBytes[i]);

}

这样写效率太低。

转自 :

C++实现的Buffer类

C++实现的CMemoryStream类

C++实现的Buffer类的更多相关文章

  1. Node.js权威指南 (5) - 使用Buffer类处理二进制数据

    5.1 创建Buffer对象 / 705.2 字符串的长度与缓存区的长度 / 725.3 Buffer对象与字符串对象之间的相互转换 / 74 5.3.1 Buffer对象的toString方法 / ...

  2. Node.js系列:Buffer类的使用

    客户端JavaScript中没有对二进制数据提供很好的支持.但是在处理TCP流或文件流时,必须要处理二进制数据.Node.js定义了一个Buffer类,用来创建一个专门存放二进制数据的缓存区. Buf ...

  3. Node.js之使用Buffer类处理二进制数据

    Node.js之使用Buffer类处理二进制数据 Buffer类可以在处理TCP流或文件流时处理二进制数据,该类用来创建一个专门存放二进制数据的缓存区. 1. 创建Buffer对象 1.1 直接创建: ...

  4. 浅析nodejs的buffer类(转)

    最近翻阅了node v0.10.4的buffer类的源代码,收获不少,也很久没有在cnode上发表文章了,想把一些收获分享给大家,有什么错误的地方希望大牛们指正啊. 前阵子有位rrestjs框架的使用 ...

  5. node.js—Buffer类(二进制数据处理模块)

    Buffer类概述 一个用于更好的操作二进制数据的类 我们在操作文件或者网络数据的时候,其实操作的就是二进制数据流 Node为我们提供了一个更加方便的去操作这种数据流的类 Buffer,他是一个全局的 ...

  6. nodeJS-使用buffer类处理二进制数据

    使用buffer类处理二进制数据 在客户端javascript脚本代码中,对于二进制数据并没有提供一个很好的支持.然后在nodejs中需要处理像TCP流或文件流时,必须要处理二进制数据.因此在node ...

  7. Buffer类的详解(转)

    Buffer 类是 java.nio 的构造基础.一个 Buffer 对象是固定数量的数据的容器,其作用是一个存储器,或者分段运输区,在这里,数据可被存储并在之后用于检索.缓冲区可以被写满或释放.对于 ...

  8. Java NIO中的Buffer类

    Buffer     缓冲,用于批量读写数据 Buffer是一个抽象类,基本数据类型都有实现类:XxxBuffer,比如ByteBuffer.CharBuffer.IntBuffer.DoubleBu ...

  9. 浅析nodejs的buffer类

    1.什么时候该用buffer,什么时候不该用 看一下如下的测试代码,分别是拼接各种不同长度的字符串,最后直接拼接了10MB的字符串 var string,string2,string3; var bu ...

随机推荐

  1. Oracle Spatial GIS相关研究

    1.Oracle Spatial 概念相关 Oracle Spatial 是Oracle 数据库强大的核心特性,包含了用于存储矢量数据类型.栅格数据类型和持续拓扑数据类型的原生数据类型.Oracle ...

  2. 把一个项目a生成后放在另一个项目b使用(b项目是例子中的ScreenWebPage_Tool)

    a项目属性---生成事件---后期生成事件命令行   xcopy /r /y  $(TargetDir)*.* $(SolutionDir)ScreenWebPage_Tool\bin\Debug\* ...

  3. Python中元组和列表

    一.list列表的操作包括以下函数: 列表操作包括以下函数: 1.cmp(list1,list2) :比较两个列表的元素 2.len(list) :列表元素个数 3.max(list) :返回列表元素 ...

  4. 排序算法lowb三人组-插入排序

    def insert_sort(li): for i in range(1, len(li)): # i表示摸到的牌的下标 tmp = li[i] # 摸到的牌 j = i - 1 while j & ...

  5. BZOJ4698: Sdoi2008 Sandy的卡片(二分 hash)

    题意 题目链接 Sol 用什么后缀数组啊 直接差分之后 二分+hash找最长公共子串就赢了啊... 时间复杂度:\(O(nlogn)\)(不过我写的是两个log..反正也能过) // luogu-ju ...

  6. QQ 聊天机器人小薇发布!

    简介 XiaoV(小薇)是一个用 Java 写的 QQ 聊天机器人 Web 服务,可以用于社群互动: 监听多个 QQ 群消息,发现有"感兴趣"的内容时通过图灵机器人进行智能回复 监 ...

  7. 准备Activiti开发环境

    1.添加jar包 在activiti-5.13 -> wars 目录下 解压 activiti-rest.war ,导入WEB-INF\lib下所有包添加到classpath中. 由于使用的是O ...

  8. 浅谈用于WEBGIS开发最重要的4个HTML5特性

    WebGIS是GIS与Internet相结合的产物,一般Internet的开发手段都可用于WEBGIS的开发,比较流行的有Javascript.FLash,到现在应该说市面上的WEBGIS产品和具有的 ...

  9. 如何使用idea把web项目打成war包

    如果是maven项目,打成war包很容易,如果是web项目,需要这样子 1. 2. 3. output directory是war包的目录 4.重新选择 第一步的操作,选择build即可.

  10. idea配置maven后提示 commond not found

    1.昨天换公司笔记本安装idea后,配置完maven后,发现自己配置的 responsity 里面没有jar包,还是在默认的.m2文件夹里面,之后发现原来是 配置的setting.xml里面的目录 这 ...