ACE (1) ACE_Message_Block
原文链接 http://www.k8s.tips/tech/2013/06/21/ACE_Message_Block/
注:以下为加速网络访问所做的原文缓存,经过重新格式化,可能存在格式方面的问题,或偶有遗漏信息,请以原文为准。
本以为ACE_Message_Block只是对void*一个简单的封装, 查看了源码发现里还有一层ACE_Data_Block:是一个带引用记数的数据区;ACE_Message_Block::duplicate会对 ACE_Message_Block进行”浅复制”,即两个ACE_Message_Block对象引用同一个ACE_Data_Block对象,但 ACE_Data_Block的引用记数为2;
参照ACE tutorial里写了一个段代码,但是发现原版本中对read_n使用方法不正确,见下漂红处:
#include <ace/Message_Block.h>
#include <ace/ACE.h>
using namespace std;
int main(int argc, char** argv)
{
ACE_Message_Block* head = new ACE_Message_Block(10);
ACE_Message_Block* mblk = head;
while (1) {
size_t nbytes = -1;
ssize_t rc = ACE::read_n(ACE_STDIN, mblk->wr_ptr(), mblk->size(), &nbytes);
if (nbytes > 0)
mblk->wr_ptr(nbytes);
if (rc <= 0)
break;
mblk->cont(new ACE_Message_Block(BUFSIZ));
mblk = mblk->cont();
}
for ( mblk = head ; mblk!=0; mblk = mblk->cont()) {
ACE::write_n(ACE_STDOUT, mblk->rd_ptr(), mblk->length());
}
head->release();
return 0;
}
这里说一下ACE::read_n 的行为:
ACE::read_n 会试图读取buf长度的数据.如果遇到文件结束(EOF)或者错误则返回 0 或 -1;如果先到达了buf长度则返回数据区长度;问题来了:如果数据读取成功,但是没有到达buf长度怎么办? 如何拿到已读数据的长度? 这就要用到ACE::read_n的第4个参数,这个参数记录了实际读取的数据长度.
在上面的code里还用到了几个函数:
ACE_Message_Block::size 指数据区的长度, 就是初始化时指定的长度,这里是10;
ACE_Message_Block::length 指数据的长度, 是 wr_ptr() – rd_ptr()的结果.
注意数据区和数据的区别….
ACE_Message_Block::cont ACE_Message_Block还实现了内存的链表结构.