java主动对象模式_ACE主动对象模式(1)

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

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

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

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

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

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

其功能代码简化如下:

class Logger: public ACE_Task

{

public:

void LogMsg(const string& msg)

{

cout<

ACE_OS::sleep(2);

}

};

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

class LogMsgCmd: public ACE_Method_Object

{

public:

LogMsgCmd(Logger *plog,const string& msg)

{

this->log=plog;

this->msg=msg;

}

int call()

{

this->log->LogMsg(msg);

return 0;

}

private:

Logger *log;

string msg;

};

class Logger: public ACE_Task

{

public:

void LogMsg(const string& msg)

{

cout<

ACE_OS::sleep(2);

}

LogMsgCmd *LogMsgActive(const string& msg)

{

new LogMsgCmd(this,msg);

}

};

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

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

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

#include

"ace/OS.h"#include

"ace/Task.h"#include

"ace/Method_Object.h"#include

"ace/Activation_Queue.h"#include

"ace/Auto_Ptr.h"#include

#include

using

namespace std;

class Logger: public ACE_Task

{

public:

Logger()

{

this->activate();

}

int svc();

void LogMsg(const string& msg);

void LogMsgActive (const string& msg);

private:

ACE_Activation_Queue cmdQueue;    //命令队列};

class LogMsgCmd: public ACE_Method_Object

{

public:

LogMsgCmd(Logger *plog,const string& msg)

{

this->log=plog;

this->msg=msg;

}

int call()

{

this->log->LogMsg(msg);

return 0;

}

private:

Logger *log;

string msg;

};

void Logger::LogMsg(const string& msg)

{

cout<

ACE_OS::sleep(2);

}

//以主动的方式记录日志void Logger::LogMsgActive(const string& msg)

{

//生成命令对象,插入到命令队列中

cmdQueue.enqueue(new LogMsgCmd(this,msg));

}

int Logger::svc()

{

while(true)

{

//遍历命令队列,执行命令

auto_ptr mo

(this->cmdQueue.dequeue ());

if (mo->call () == -1)

break;

}

return 0;

}

int main (int argc, ACE_TCHAR *argv[])

{

Logger log;

log.LogMsgActive ("hello");

ACE_OS::sleep(1);

log.LogMsgActive("abcd");

while(true)

ACE_OS::sleep(1);

return 0;

}

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

主动对象的基本结构就是这样,然而,由于主动对象是异步调用的,又引出了如下两个新问题:方法调用线程如何知道该方法已经执行完成?

如何或得方法的返回值?

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

课程目录 1001_RocketMQ_简介[免费观看]17:13 2002_RocketMQ_核心概念详解[免费观看]36:44 3003_RocketMQ_集群构建模型详解(一)27:47 4004_RocketMQ_集群构建模型详解(二)23:36 5005_RocketMQ_双主模式集群环境搭建39:05 6006_RocketMQ_控制台使用讲解10:41) 7007_RocketMQ_Broker配置文件详解20:38 8008_RocketMQ_helloworld示例讲解34:29 9009_RocketMQ_整体架构概述详解15:36 10010_RocketMQ_Producer_API详解24:49 11011_RocketMQ_Producer_顺序消费机制详解24:37 12012_RocketMQ_Producer_事务消息机制详解38:15 13013_RocketMQ_Consumer_Push和Pull模式及使用详解28:21 14014_RocketMQ_Consumer_配置参数详解06:45 15015_RocketMQ_Consumer_重试策略详解33:08 16016_RocketMQ_Consumer_幂等去重策略详解33:48 17017_RocketMQ_消息模式及使用讲解14:12 18018_RocketMQ_双主双从集群环境搭建与使用详解40:30 19019_RocketMQ_FilterServer机制及使用详解35:31 20020_RocketMQ_管理员命令16:37 JAVA-ACE-架构师系列视频课程- RocketMQ(下)订单实战视频课程 适用人群: 高级java工程师、java架构师 共23课时共9小时33分钟更新时间:2017-04-05 课程目标 RocketMQ(下)订单实战主要讲解rmq的分布式实战项目,围绕着订单模块等进行讲解,实现补偿等 课程目录 101_rocketmq_实战项目介绍[免费观看]22:54 202_rocketMQ实战项目设计(一)26:36 303_rocketMQ实战项目设计(二)18:39 404_rocketMQ实战-环境搭建(一)27:03 505_rocketMQ实战-环境搭建(二)28:24 606_rocketMQ实战-生产者与spring结合30:20 707_rocketMQ实战-消费者与spring结合25:47 808_rocketMQ实战-数据库模型设计28:18 909_rocketMQ实战-数据库DAO代码生成27:26 1010_rocketMQ实战-远程RPC接口设计与实现(一)29:29 1111_rocketMQ实战-远程RPC接口设计与实现(二)24:03 1212_rocketMQ实战-远程RPC接口设计与实现(三)16:44 1313_rocketMQ实战-下单流程(一)21:57 1414_rocketMQ实战-下单流程(二)22:51 1515_rocketMQ实战-下单流程(三)33:20 1616_rocketMQ实战-下单流程(四)27:47 1717_rocketMQ实战-下单流程(五)23:25 1818_rocketMQ实战-下单流程(六)17:48 1919_rocketMQ实战-下单流程(七)26:56 2020_rocketMQ实战-下单流程(八)-商品库存25:43 2121_rocketMQ实战-下单流程(九)-商品库存28:47 2222_rocketMQ实战-下单流程(十)-支付模块11:11 2323_rocketMQ实战-整体联调27:44
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值