ACE-Streams架构简介及应用
一概述
Streams框架是管道和过滤构架模式的一种实现,主要应用于处理数据流的系统。其实现以Task框架为基础。Task框架有两个特性非常适用于Streams框架:一是Task框架可用于创建独立线程的并发环境,这适合应用于ACE Streams框架中的主动过滤器;二是Task框架有统一的数据传输结果——消息队列,这适用于Streams框架中的管道。
二ACE_Task类
这里主要介绍与Streams框架相关的部分。
成员变量
 Task_T.h
 class ACE_Task : public ACE_Task_Base
 {
 /// Queue of messages on the ACE_Task..
   ACE_Message_Queue<ACE_SYNCH_USE, TIME_POLICY> *msg_queue_;
   /// true if should delete Message_Queue, false otherwise.
   bool delete_msg_queue_;
   /// Back-pointer to the enclosing module.
   ACE_Module<ACE_SYNCH_USE, TIME_POLICY> *mod_;
   /// Pointer to adjacent ACE_Task.
   ACE_Task<ACE_SYNCH_USE, TIME_POLICY> *next_;
 }
put函数
在ACE_Task框架中,put函数没有实际作用,在默认情况下,该函数没有执行任何操作,仅仅返回0.但是在Streams框架中,put函数与put_next函数结合起来可以实现数据在过滤器间的传输。如果put函数将数据保存在消息队列中,通过独立的线程来处理这些消息,那么它将成为一个主动过滤器;反之,如果put函数直接对数据进行处理,然后交给下一个过滤器,那么它就是一个被动过滤器。
 Task.cpp
 /// Default ACE_Task put routine.
 int
 ACE_Task_Base::put (ACE_Message_Block *, ACE_Time_Value *)
 {
   ACE_TRACE ("ACE_Task_Base::put");
   return ;
 }
put_next函数
如果在数据处理流水线有下一个过滤器,那么put_next函数用于将数据交给下一个过滤器处理。如下:
 Task_T.inl
 // Transfer msg to the next ACE_Task.
 template <ACE_SYNCH_DECL, class TIME_POLICY> ACE_INLINE int
 ACE_Task<ACE_SYNCH_USE, TIME_POLICY>::put_next (ACE_Message_Block *msg, ACE_Time_Value *tv)
 {
   ACE_TRACE ("ACE_Task<ACE_SYNCH_USE, TIME_POLICY>::put_next");
   return this->next_ ==  ? - : this->next_->put (msg, tv);
 }
next_指向的是有序过滤器的下一个过滤器。通过put_next函数,可以将数据交给下一个过滤器处理。
Streams框架应用示例
在这个示例中,我们将一个数据流的处理分为4步,在Streams框架中,将每个处理步骤称为一个Module:
- Logrec_Reader:从文件中读取记录,然后交给下一个步骤。
- Logrec_Timer:在记录尾部加上“format_data”
- Logrec_Suffix:在记录尾部加上一个后缀——suffix
- Logrec_Write:将记录显示在终端上
Logrec_Reader类
其是ACE_Task的子类,是一个主动对象类,有独立的控制线程,线程处理函数是svc。在Streams框架中,Logrec_Reader类是一个主动过滤器,代码如下:
class Logrec_Reader : public ACE_Task<ACE_MT_SYNCH>
{
private:
ifstream fin; //标准输入流
public:
Logrec_Reader(ACE_TString logfile)
{
fin.open(logfile.c_str()); //ACE_TString.c_str() 转换为char
}
virtual int open (void *)
{
return activate();
} virtual int svc()
{
ACE_Message_Block *mblk;
int len = ;
const int LineSize = ;
char file_buf[LineSize]; while(!fin.eof())
{
fin.getline(file_buf, LineSize);
len = ACE_OS::strlen(file_buf);
ACE_NEW_RETURN(mblk, ACE_Message_Block(len + ), );
if (file_buf[len - ] == '\r')
{
len = len - ;
}
mblk->copy(file_buf, len);
// 通过put_next函数,将消息传递给下一个过滤器
put_next(mblk);
}
//构造一个MB_STOP消息
ACE_NEW_RETURN(mblk, ACE_Message_Block (, ACE_Message_Block::MB_STOP), );
put_next(mblk);
fin.close();
ACE_DEBUG((LM_DEBUG, "read svc return. \n"));
return ;
}
};
Logrec_Timer类
也是ACE_Task的子类,但是它不是主动对象类,没有创建独立线程。其实现了put函数,这个函数被它的上一个过滤器(Logrec_Reader)调用,并且数据直接在这个函数中处理。
这里for循环用于处理消息链表,在示例中并没有使用链表因此for循环只会执行一次。
class Logrec_Timer : public ACE_Task<ACE_SYNCH>
{
private:
void format_data(ACE_Message_Block *mblk)
{
char *msg = mblk->data_block()->base();
strcat(msg, "format_data");
}
public:
virtual int put(ACE_Message_Block *mblk, ACE_Time_Value *)
{
for (ACE_Message_Block *temp = mblk;
temp != ; temp = temp->cont())
{
if (temp->msg_type() != ACE_Message_Block::MB_STOP)
{
format_data(temp);
}
}
return put_next(mblk);
}
};
Logrec_Suffix类
类似Logrec_Timer
class Logrec_Suffix : public ACE_Task<ACE_SYNCH>
{
public:
void suffix(ACE_Message_Block *mblk)
{
char *msg = mblk->data_block()->base();
strcat(msg, "suffix\n");
}
virtual int put(ACE_Message_Block *mblk, ACE_Time_Value *)
{
for (ACE_Message_Block *temp = mblk;
temp != ; temp = temp->cont())
{
if (temp->msg_type() != ACE_Message_Block::MB_STOP)
{
suffix(temp);
}
}
return put_next(mblk);
}
};
Logrec_Write类
这里put函数由上一个过滤器(Logrec_Suffix)调用,其并没有对数据进行实际处理,只是将数据放入队列中,由线程独立处理。
class Logrec_Write : public ACE_Task<ACE_SYNCH>
{
public:
virtual int open(void*)
{
ACE_DEBUG((LM_DEBUG, "Logrec_Writer.\n"));
return activate();
} virtual int put (ACE_Message_Block *mblk, ACE_Time_Value *to)
{
return putq(mblk, to);
} virtual int svc()
{
int stop = ;
for (ACE_Message_Block *mb; !stop && getq(mb) != -;)
{
if (mb->msg_type() == ACE_Message_Block::MB_STOP)
{
stop = ;
}
else{
ACE_DEBUG((LM_DEBUG, "%s", mb->base()));
}
put_next(mb);
}
return ;
}
};
Main
这里讲ACE_Module放入Streams中,ACE_Module才是真正数据处理的Module。ACE_Streams类有两个数据成员:stream_head_和stream_tail_,它们指向ACE_Module链表的首尾。对于每个Streams,其默认带有首尾两个Module,而后可以通过push将数据处理的Module放入执行链表中。每个Module包含两个Task,分别为读Task和写Task。在示例中仅注册了写Task,这些Module和Task通过next指针构成一个有序的串。
这里注意push有顺序要求,最后push即栈顶的为先执行的Module。
int ACE_TMAIN(int argc, ACE_TCHAR *argv[])
{
if (argc != )
{
ACE_ERROR_RETURN((LM_ERROR, "usage: %s logfile\n", argv[]), );
} ACE_TString logfile (argv[]); ACE_Stream<ACE_SYNCH> stream; ACE_Module<ACE_MT_SYNCH> *module[];
module[] = new ACE_Module<ACE_MT_SYNCH>("Reader", new Logrec_Reader(logfile), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Formatter", new Logrec_Timer(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Separator", new Logrec_Suffix(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Writer", new Logrec_Write(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER); for ( int i = ; i >= ; --i )
{
if (stream.push(module[i]) == -)
{
ACE_ERROR_RETURN((LM_ERROR, "push %s module into stream failed.\n", module[i]->name()), );
}
ACE_DEBUG((LM_DEBUG, "push %s module into stream success. \n", module[i]->name()));
}
ACE_Thread_Manager::instance()->wait();
}
附完整代码及结果:https://github.com/ShiningZhang/ACE_Learning/tree/master/streams
/*************************************************************************
> File Name: logrec.cpp
> Author:
> Mail:
> Created Time: Fri 13 Oct 2017 04:19:39 PM CST
************************************************************************/
#include <fstream>
#include <ace/Synch.h>
#include <ace/Task.h>
#include <ace/Message_Block.h>
#include <ace/Stream.h>
#include "ace/Thread_Manager.h"
#include <ace/Time_Value.h>
#include <ace/Module.h> using namespace std; class Logrec_Reader : public ACE_Task<ACE_MT_SYNCH>
{
private:
ifstream fin; //标准输入流
public:
Logrec_Reader(ACE_TString logfile)
{
fin.open(logfile.c_str()); //ACE_TString.c_str() 转换为char
}
virtual int open (void *)
{
return activate();
} virtual int svc()
{
ACE_Message_Block *mblk;
int len = ;
const int LineSize = ;
char file_buf[LineSize]; while(!fin.eof())
{
fin.getline(file_buf, LineSize);
len = ACE_OS::strlen(file_buf);
ACE_NEW_RETURN(mblk, ACE_Message_Block(len + ), );
if (file_buf[len - ] == '\r')
{
len = len - ;
}
mblk->copy(file_buf, len);
// 通过put_next函数,将消息传递给下一个过滤器
put_next(mblk);
}
//构造一个MB_STOP消息
ACE_NEW_RETURN(mblk, ACE_Message_Block (, ACE_Message_Block::MB_STOP), );
put_next(mblk);
fin.close();
ACE_DEBUG((LM_DEBUG, "read svc return. \n"));
return ;
}
}; class Logrec_Timer : public ACE_Task<ACE_SYNCH>
{
private:
void format_data(ACE_Message_Block *mblk)
{
char *msg = mblk->data_block()->base();
strcat(msg, "format_data");
}
public:
virtual int put(ACE_Message_Block *mblk, ACE_Time_Value *)
{
for (ACE_Message_Block *temp = mblk;
temp != ; temp = temp->cont())
{
if (temp->msg_type() != ACE_Message_Block::MB_STOP)
{
format_data(temp);
}
}
return put_next(mblk);
}
}; class Logrec_Suffix : public ACE_Task<ACE_SYNCH>
{
public:
void suffix(ACE_Message_Block *mblk)
{
char *msg = mblk->data_block()->base();
strcat(msg, "suffix\n");
}
virtual int put(ACE_Message_Block *mblk, ACE_Time_Value *)
{
for (ACE_Message_Block *temp = mblk;
temp != ; temp = temp->cont())
{
if (temp->msg_type() != ACE_Message_Block::MB_STOP)
{
suffix(temp);
}
}
return put_next(mblk);
}
}; class Logrec_Write : public ACE_Task<ACE_SYNCH>
{
public:
virtual int open(void*)
{
ACE_DEBUG((LM_DEBUG, "Logrec_Writer.\n"));
return activate();
} virtual int put (ACE_Message_Block *mblk, ACE_Time_Value *to)
{
return putq(mblk, to);
} virtual int svc()
{
int stop = ;
for (ACE_Message_Block *mb; !stop && getq(mb) != -;)
{
if (mb->msg_type() == ACE_Message_Block::MB_STOP)
{
stop = ;
}
else{
ACE_DEBUG((LM_DEBUG, "%s", mb->base()));
}
put_next(mb);
}
return ;
}
}; int ACE_TMAIN(int argc, ACE_TCHAR *argv[])
{
if (argc != )
{
ACE_ERROR_RETURN((LM_ERROR, "usage: %s logfile\n", argv[]), );
} ACE_TString logfile (argv[]); ACE_Stream<ACE_SYNCH> stream; ACE_Module<ACE_MT_SYNCH> *module[];
module[] = new ACE_Module<ACE_MT_SYNCH>("Reader", new Logrec_Reader(logfile), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Formatter", new Logrec_Timer(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Separator", new Logrec_Suffix(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER);
module[] = new ACE_Module<ACE_MT_SYNCH>("Writer", new Logrec_Write(), , , ACE_Module<ACE_SYNCH>::M_DELETE_READER); for ( int i = ; i >= ; --i )
{
if (stream.push(module[i]) == -)
{
ACE_ERROR_RETURN((LM_ERROR, "push %s module into stream failed.\n", module[i]->name()), );
}
ACE_DEBUG((LM_DEBUG, "push %s module into stream success. \n", module[i]->name()));
}
ACE_Thread_Manager::instance()->wait();
}


test:

ACE-Streams架构简介及应用的更多相关文章
- LoadRunner系统架构简介
		1.LoadRunner系统架构简介 LoadRunner是通过创建虚拟用户来代替真实实际用户来操作客户端软件比如Internet Explorer,来向IIS.Apache等Web服务器发送HTTP ... 
- crtmpserver的架构简介
		crtmpserver的架构简介 一.层 Layers . 机器层 Machine layer . 操作系统层 Operating System Layer This layer is compo ... 
- Extjs6官方文档译文——应用架构简介(MVC,MVVM)
		应用架构简介 Extjs 同时提供对于MVC和MVVM应用架构的支持.这两个架构方式共享某些概念,而且都旨在沿着逻辑层面划分应用程序代码.每种方法在选择如何划分应用组件上都有其各自的优势. 本指南的目 ... 
- Kafka:架构简介【转】
		转:http://www.cnblogs.com/f1194361820/p/6026313.html Kafka 架构简介 Kafka是一个开源的.分布式的.可分区的.可复制的基于日志提交的发布订阅 ... 
- LoadRunner系统架构简介与运行原理
		1.LoadRunner系统架构简介 LoadRunner是通过创建虚拟用户来代替真实实际用户来操作客户端软件比如Internet Explorer,来向IIS.Apache等Web服务器发送HTTP ... 
- Cortex-M0(+)内核的处理器架构简介
		Cortex-M0(+)内核的处理器架构简介 2015年03月02日 16:51:12 阅读数:3158 系统架构 Cortex-M0处理器具有32位系统总线接口,以及32位地址线,即有4GB的地址空 ... 
- 【转帖】LoadRunner系统架构简介
		LoadRunner系统架构简介: LoadRunner是通过创建虚拟用户来代替真实实际用户来操作客户端软件比如Internet Explorer,来向IIS.Apache等Web服务器发送HTTP协 ... 
- TP 5.0 架构 简介
		TP 5.0 架构 简介 thinkphp 5.0 为API开发而设计的的高性能框架,是与以往thinkphp5.0以下版本大不相同的新型框架,病对以颠覆和重构版本,采用全新的架构思想,引入了更多的P ... 
- ceph架构简介
		ceph架构简介 在测试OpenStack的后端存储时,看到了ceph作为后端存储时的各种优势 ,于是查询资料,总结了这篇ceph架构的博客,介绍了ceph的架构和ceph的核心组件.ceph整体十分 ... 
随机推荐
- 禁用substr、substring、mid函数的sql注入脚本
			#encodeing=utf-8 import requests import sys reload(sys) sys.setdefaultencoding('utf-8') payloads = l ... 
- python开发_difflib字符串比较
			在python的difflib中 HtmlDiff:比较后以html方法展示 我们比较的是字符串: 'hello world!' 和 'hElLO Wor2d!' 具体代码: from difflib ... 
- [转]Android:异步处理之AsyncTask的应用(二)
			2014-11-07 既然UI老人家都这么忙了,我们这些开发者肯定不能不识趣的去添乱阻塞UI线程什么的,否则UI界面万一停止响应了呢——这不是招骂的节奏么?!所以我们知道用Handler+Th ... 
- 蓝屏代码详解(更新WIN7蓝屏代码)
			6位代码含意 0 0x0000 作业完成. 1 0x0001 不正确的函数. 2 0x0002 系统找不到指定的档案. 3 0x0003 系统找不到指定的路径. 4 0x0004 系统无法开启 ... 
- Nagios 监控mysqlserver具体实现过程
			,之后在页面就能够看到监控效果了 參考文章:http://os.51cto.com/art/201409/452605.htm 
- this与JavaScrip中的四种调用模式
			this是什么 方法调用模式 构造器调用模式 函数调用模式 apply/call模式 this是什么 —In most languages, ‘this’ is a reference to the ... 
- 根据url下载图片和页面
			需要将&tp=webp&wxfrom=5去掉,既可以在任何地方显示,也可以下载了 http://mmbiz.qpic.cn/mmbiz_jpg/bf8pC39RBhGFOH1ib9Ac ... 
- UIControl的子类UISwitch, UISegmentedCntrol, UIPageControl详解
			- (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typica ... 
- PAT 1033. To Fill or Not to Fill (贪心)
			PAT-A的最后一题,最终做出来了... 是贪心,通过局部最优获得全局最优. 1. 将加油站按距离升序排序 2. 记录当前所在的加油站index,存有的汽油,花费.向后遍历全部 该站可抵达的加油站 3 ... 
- Table里td中的文本过长,设置不换行,随内容同行显示(转载)
			当td中内容过长时,内容会溢出,换行显示,美观超级差,在td里设置这个属性 "white-space:nowrap 就可以解决排版问题啦 <td style="white ... 
