ACE主动对象模式(1)

转载于:http://www.cnblogs.com/TianFang/archive/2006/12/11/589168.html

主动对象模式用于降低方法执行和方法调用之间的耦合。该模式描述了另外一种更为透明的任务间通信方法。

传统上,所有的对象都是被动的代码段,对象中的代码是在对它发出方法调用的线程中执行的,当方法被调用时,调用线程将阻塞,直至调用结束。而主动对象却不一样。这些对象具有自己的命令执行线程,主动对象的方法将在自己的执行线程中执行,不会阻塞调用方法。

例如,设想对象"A"已在你的程序的main()函数中被实例化。当你的程序启动时,OS创建一个线程,以从main()函数开始执行。如果你调用对象A的任何方法,该线程将"流过"那个方法,并执行其中的代码。一旦执行完成,该线程返回调用该方法的点并继续它的执行。但是,如果"A"是主动对象,事情就不是这样了。在这种情况下,主线程不会被主动对象借用。相反,当"A"的方法被调用时,方法的执行发生在主动对象持有的线程中。另一种思考方法:如果调用的是被动对象的方法(常规对象),调用会阻塞(同步的);而另一方面,如果调用的是主动对象的方法,调用不会阻塞(异步的)。

由于主动对象的方法调用不会阻塞,这样就提高了系统响应速度,在网络编程中是大有用武之地的。

在这里我们将一个"Logger"(日志记录器)对象对象为例来介绍如何将一个传统对象改造为主动对象,从而提高系统响应速度。

Logger的功能是将一些系统事件的记录在存储器上以备查询,由于Logger使用慢速的I/O系统来记录发送给它的消息,因此对Logger的操作将会导致系统长时间的等待。

其功能代码简化如下:

1 class Logger: public ACE_Task<ACE_MT_SYNCH>
2 {
3 public:
4     void LogMsg(const string& msg)
5     {
6         cout<<endl<<msg<<endl;
7         ACE_OS::sleep(2);
8     }
9 };

为了实现实现记录日志操作的主动执行,我们需要用命令模式将其封装,从而使得记录日志的方法能在合适的时间和地方主动执行,封装方式如下:

 1 class LogMsgCmd: public ACE_Method_Object
 2 {
 3 public:
 4     LogMsgCmd(Logger *plog,const string& msg)
 5     {
 6         this->log=plog;
 7         this->msg=msg;
 8     }
 9
10     int call()
11     {
12         this->log->LogMsg(msg);
13         return 0;
14     }
15
16 private:
17     Logger *log;
18     string msg;
19 };
20
21 class Logger: public ACE_Task<ACE_MT_SYNCH>
22 {
23 public:
24     void LogMsg(const string& msg)
25     {
26         cout<<endl<<msg<<endl;
27         ACE_OS::sleep(2);
28     }
29
30     LogMsgCmd *LogMsgActive(const string& msg)
31     {
32         new LogMsgCmd(this,msg);
33     }
34 };

这里对代码功能做一下简单的说明:

ACE_Method_Object是ACE提供的命令模式借口,命令接口调用函数为int call(),在这里通过它可以把每个操作日志的调用封装为一个LogMsgCmd对象,这样,当原来需要调用LogMsg的方法的地方只要调用LogMsgActive即可生成一个LogMsgCmd对象,由于调用LogMsgActive方法,只是对命令进行了封装,并没有进行日志操作,所以该方法会立即返回。然后再新开一个线程,将LogMsgCmd对象作为参数传入,在该线程中执行LogMsgCmd对象的call方法,从而实现无阻塞调用。

然而,每次对一个LogMsg调用都开启一个新线程,无疑是对资源的一种浪费,实际上我们往往将生成的LogMsgCmd对象插入一个命令队列中,只新开一个命令执行线程依次执行命令队列中的所有命令。并且,为了实现对象的封装,命令队列和命令执行线程往往也封装到Logger对象中,代码如下所示:

 1 #include "ace/OS.h"
 2 #include "ace/Task.h"
 3 #include "ace/Method_Object.h"
 4 #include "ace/Activation_Queue.h"
 5 #include "ace/Auto_Ptr.h"
 6
 7 #include <string>
 8 #include <iostream>
 9 using namespace std;
10
11 class Logger: public ACE_Task<ACE_MT_SYNCH>
12 {
13 public:
14     Logger()
15     {
16         this->activate();
17     }
18
19     int svc();
20     void LogMsg(const string& msg);
21     void LogMsgActive (const string& msg);
22
23 private:
24     ACE_Activation_Queue cmdQueue;    //命令队列
25 };
26
27 class LogMsgCmd: public ACE_Method_Object
28 {
29 public:
30     LogMsgCmd(Logger *plog,const string& msg)
31     {
32         this->log=plog;
33         this->msg=msg;
34     }
35
36     int call()
37     {
38         this->log->LogMsg(msg);
39         return 0;
40     }
41
42 private:
43     Logger *log;
44     string msg;
45 };
46
47 void Logger::LogMsg(const string& msg)
48 {
49     cout<<endl<<msg<<endl;
50     ACE_OS::sleep(2);
51 }
52
53 //以主动的方式记录日志
54 void Logger::LogMsgActive(const string& msg)
55 {
56     //生成命令对象,插入到命令队列中
57     cmdQueue.enqueue(new LogMsgCmd(this,msg));
58 }
59
60 int Logger::svc()
61 {
62     while(true)
63     {
64         //遍历命令队列,执行命令
65         auto_ptr<ACE_Method_Object> mo
66             (this->cmdQueue.dequeue ());
67
68         if (mo->call () == -1)
69             break;
70     }
71     return 0;
72 }
73
74 int main (int argc, ACE_TCHAR *argv[])
75 {
76     Logger log;
77     log. LogMsgActive ("hello");
78
79     ACE_OS::sleep(1);
80     log.LogMsgActive("abcd");
81
82     while(true)
83         ACE_OS::sleep(1);
84
85     return 0;
86 }

在这里需要注意一下命令队列ACE_Activation_Queue对象,它是线程安全的,使用方法比较简单,这里我也不多介绍了。

主动对象的基本结构就是这样,然而,由于主动对象是异步调用的,又引出了如下两个新问题:

  1. 方法调用线程如何知道该方法已经执行完成?
  2. 如何或得方法的返回值?

这两个问题将在下回给与解决。

时间: 2024-09-27 14:14:41

ACE主动对象模式(1)的相关文章

ACE主动对象模式(2)

转载于:http://www.cnblogs.com/TianFang/archive/2006/12/12/590153.html 我们简单的介绍了一下ACE主动对象实现方式,同时提出了两个问题: 方法调用线程如何知道该方法已经执行完成? 如何或得方法的返回值? 要解决这两个问题,首先得介绍一下ACE_Future对象,ACE_Future是表示一个会在将来被赋值的"期货"对象,可以通过ready()函数查询它是否已经被赋值.该对象创建的时候是未赋值的,后期可以通过set()函数来进

微信企业号开发一:主动调用模式之TOKEN申请

微信企业号开发,数据访问格式分为以下三种: 1.企业应用调用企业号提供的接口,管理或查询企业号后台所管理的资源.或给成员发送消息等,以下称主动调用模式. 2.企业号把用户发送的消息或用户触发的事件推送给企业应用,由企业应用处理,以下称回调模式. 3.用户在微信中阅读企业应用下发的H5页面,该页面可以调用微信提供的原生接口,使用微信开放的终端能力,以下称JSAPI模式: 官方文档地址:http://qydev.weixin.qq.com/wiki/index.php 现在我们来看一下主动调用模式的

空对象模式(Null Object Pattern)

空对象模式:用一个空对象来取代null实例的检查,空对象实现一个不做任何动作的关系.(消除如if(Object == null) 这样的检查null实例代码) 例子: public abstract class Duck { public String name; public abstract void getName(); } public class ConcreteDuck extends Duck { public ConcreteDuck(String name) { this.na

C# 设计模式之空对象模式

最近看了不少的书籍和视频等相关资料,决定自己边学习边写一下个人对设计模式的理解,如果有不对的请大家多多指正. 今天先说说我个人觉得最简单的设计模式 -- [空对象模式] 空对象模式可以减少客户端对对象判断是否为空的重复操作,也可以减少异常发生的情况[如果客户端调用程序忘记了检查对象是否为空,而直接访问对象中的数据,则有可能会报出异常,影响程序运行]. 废话不说了,先上我写的代码,下面是未运用空对象模式之前的示例代码: 要返回的实体代码,Person.cs: namespace NullObjec

主动对象

主动对象那么到底什么是主动对象呢?传统上,所有的对象都是被动的代码段,对象中的代码是在对它发出方法调用的线程中执行的.也就是,调用线程(calling threads)被“借出”,以执行被动对象的方法.而主动对象却不一样.这些对象持有它们自己的线程(甚或多个线程),并将这个线程用于执行对它们的任何方法的调用.因而,如果你想象一个传统对象,在里面封装了一个线程(或多个线程),你就得到了一个主动对象.例如,设想对象“A”已在你的程序的main()函数中被实例化.当你的程序启动时,OS创建一个线程,以

空对象模式

在学习Head First设计模式中的“命令模式”过程中,偶然发现可以用在coding过程中的小技巧.赶紧记录,以备后用! 具体可以称之为“空对象”模式,而且专门用来处理对象为null的情形. 比如以下情形: Command接口: public interface Command { public void execute(); } 测试代码: Command[] commands = new Command[10]; // initial commands[0] and the others

角色对象模式

意图 单个对象透过不同的角色对象来满足不同客户的不同需求.每一个角色对象针对不同的客户内容来扮演其角色.对象能够动态的管理其角色集合.角色作为独立的对象是的不同的内容能够简单的被分离开来,系统间的配置也变得容易. 译注:为了行文的流畅性及内容意思的准确性,尽量贴近原文使用英文单词标记特定内容, 如Customer表示客户,Client表示客户端,Component表示组件等.因为有各种图例说明,所以在图例说明时,使用原题中的英文单词对应图中内容.有时也中英文交叉使用.因为网页显示的问题,中文黑体

【设计模式】空对象模式

在空对象模式(Null Object Pattern)中,一个空对象取代 NULL 对象实例的检查.Null 对象不是检查空值,而是反应一个不做任何动作的关系.这样的 Null 对象也可以在数据不可用的时候提供默认的行为. 在空对象模式中,我们创建一个指定各种要执行的操作的抽象类和扩展该类的实体类,还创建一个未对该类做任何实现的空对象类,该空对象类将无缝地使用在需要检查空值的地方. 实现 我们将创建一个定义操作(在这里,是客户的名称)的 AbstractCustomer 抽象类,和扩展了 Abs

【VBA研究】工作表自己主动筛选模式检測

作者:iamlaosong 用VBA程序处理Excel数据文件.用户的数据文件有时处于自己主动筛选模式,往往导致数据处理不对.为此,须要检測工作表是否处于该模式,假设是,则去掉自己主动筛选.语句例如以下: If ActiveSheet.AutoFilterMode = True Then Selection.AutoFilter 这个语句一般仅仅用于去掉自己主动筛选,尽管Selection.AutoFilter也能够加上自己主动筛选,但筛选位置却可能在当前单元格处,所以要注意.加自己主动筛选前,