在这里,我们先定义一下,什么是并发对象。不同于一般的对象,并发对象指的是该对象方法的调用与方法的执行不在同一个线程內,也即:该对象方法被异步执行。这其实是在多线程编程环境下典型的计算特征,线程引入了异步。从另一个角度来看,并发对象其
实是面向对象的概念应用于多线程计算环境下的产物。
我们将从以下几个方面来讨论 Active Object 模式。
我们都知道,基于多线程机制的并发编程模式可以极大提高应用的 QoS(Quality of Service)。典型的例子如,我们在开发服务器端的应用时,通行的做法就是通过多线程机制并发地服务客户端提交上来的请求,以期提高服务器对客户端的反应度 (Responsiveness)。同时,相比于单线程的应用,并发的多线程应用相对要复杂得多。在多线程的计算环境里,并发对象被所有的调用者线程所共享。一般来说,并发对象的设计实现需要考虑下面的几个重要因素:
我们使用 Active Object 设计模式来解决这些问题。
Active Object 设计模式的本质是解耦合方法的调用 (Method invocation) 与方法的执行 (Method execution),方法调用发生在调用者线程上下文中,而方法的执行发生在独立于调用者线程的 Active Object 线程上下文中。并且重要的一点是,该方法与其它普通的对象成员方法对于调用者来说,没有什么特别的不同。从运行时的角度来看,这里涉及到两类线程,一个是调用者线程,另外一个是 Active Object 线程,我们会在下面更详细地谈到。
在 Active Object 模式中,主要有以下几种类型的参与者:
在 Active Object 设计模式中,在参与者之间将发生如下的协作过程:
从图 1 我们可以看到,步骤 1 到步骤 6 运行在调用者线程中,而步骤 7 到步骤 12 运行在 Active Object 的线程中。
在本节中,我们给出 Active Object 的 C++ 示例实现。
调用者调用 Proxy 的 get() 方法,从 Active Object 获得 Message。我们可以假定,在真实的应用中, get() 方法的实现受制于某些慢速的 IO 操作,比如需要通过 TCP Socket 从远端的机器获得 Message, 然后返回给调用者。所以我们使用 Active Object 来实现该应用,通过线程的并发达到提高应用的 QoS。
class MQ_Servant { public: // Constructor and destructor. MQ_Servant (size_t mq_size); virtual ~MQ_Servant (); // Message queue implementation operations. void put (const Message &msg); Message get (); // Predicates. bool empty () const; bool full () const; private: // Internal queue representation, e.g., a circular // array or a linked list, that does not use any // internal synchronization mechanism. };
MQ_Servant 是真正的服务提供者,实现了 Proxy 中定义的方法。put() 和 get() 方法用来操作底层的队列。另外,Servant 的实现是纯粹的应用逻辑实现,或者称为商业逻辑实现,没有混合任何的线程同步机制 , 这有利于我们进行应用逻辑的重用,而不需要考虑不同的线程同步机制。
class MQ_Proxy { public: // Bound the message queue size. enum { MQ_MAX_SIZE = 100 }; MQ_Proxy (size_t size = MQ_MAX_SIZE) :scheduler_ (size), servant_ (size) { } // Schedule <put> to execute on the active object. void put (const Message &msg) { Method_Request *mr = new Put(servant_,msg); scheduler_.insert (mr); } // Return a <Message_Future> as the "future" result of // an asynchronous <get> method on the active object. Message_Future get () { Message_Future result; Method_Request *mr = new Get (&servant_,result); scheduler_.insert (mr); return result; } // empty() and full() predicate implementations ... private: // The servant that implements the active object // methods and a scheduler for the message queue. MQ_Servant servant_; MQ_Scheduler scheduler_; };
同一个进程中的多个调用者线程可以共享同一个 Proxy。
class Method_Request { public: // Evaluate the synchronization constraint. virtual bool can_run () const = 0 // Execute the method. virtual void call () = 0; }; // Inherites from Method_Request class Get : public Method_Request { public: Get (MQ_Servant *rep, const Message_Future &f) :servant_ (rep), result_ (f) { } virtual bool can_run () const { // Synchronization constraint: cannot call a // <get> method until queue is not empty. return !servant_->empty (); } virtual void call () { // Bind dequeued message to the future result. result_ = servant_->get (); } private: MQ_Servant *servant_; Message_Future result_; };
class Activation_List { public: // Block for an "infinite" amount of time waiting // for <insert> and <remove> methods to complete. enum { INFINITE = -1 }; // Define a "trait". typedef Activation_List_Iterator iterator; Activation_List (); // Insert <method_request> into the list, waiting up // to <timeout> amount of time for space to become // available in the queue. Throws the <System_Ex> // exception if <timeout> expires. void insert (Method_Request *method_request,Time_Value *timeout = 0); // Remove <method_request> from the list, waiting up // to <timeout> amount of time for a <method_request> // to be inserted into the list. Throws the // <System_Ex> exception if <timeout> expires. void remove (Method_Request *&method_request, Time_Value *timeout = 0); private: // Synchronization mechanisms, e.g., condition // variables and mutexes, and the queue implementation, // e.g., an array or a linked list, go here. };
Activation List 的实际上就是一个线程同步机制保护下的 Method Request 队列,对该队列的所有操作 (insert/remove) 都应该是线程安全的。从本质上讲,Activation List 所基于的就是典型的生产者 / 消费者并发编程模型,调用者线程作为生产者把 Method Request 放入该队列,Active Object 线程作为消费者从该队列拿出 Method Request, 并执行。
class MQ_Scheduler { public: // Initialize the <Activation_List> and make <MQ_Scheduler> // run in its own thread of control. // we call this thread as Active Object thread. MQ_Scheduler () : act_list_() { // Spawn separate thread to dispatch method requests. // The following call is leveraging the parallelism available on native OS // transparently Thread_Manager::instance ()->spawn (&svc_run,this); } // ... Other constructors/destructors, etc. // Put <Method_Request> into <Activation_List>. This // method runs in the thread of its client,i.e. // in the proxy's thread. void insert (Method_Request *mr) { act_list_.insert (mr); } // Dispatch the method requests on their servant // in its scheduler's thread of control. virtual void dispatch () { // Iterate continuously in a separate thread(Active Object thread). for (;;) { Activation_List::iterator request; // The iterator's <begin> method blocks // when the <Activation_List> is empty. for(request = act_list_.begin (); request != act_list_.end ();++request){ // Select a method request whose // guard evaluates to true. if ((*request).can_run ()) { // Take <request> off the list. act_list_.remove (*request); (*request).call () ; delete *request; } // Other scheduling activities can go here, // e.g., to handle when no <Method_Request>s // in the <Activation_List> have <can_run> // methods that evaluate to true. } } } private: // List of pending Method_Requests. Activation_List act_list_; // Entry point into the new thread. static void *svc_run (void *arg) { MQ_Scheduler *this_obj = static_cast<MQ_Scheduler *> (args); this_obj->dispatch (); } };
class Message_Future { public: // Initializes <Message_Future> to // point to <message> immediately. Message_Future (const Message &message); //Other implementatio…… // Block upto <timeout> time waiting to obtain result // of an asynchronous method invocation. Throws // <System_Ex> exception if <timeout> expires. Message result (Time_Value *timeout = 0) const; private: //members definition here…… };
事实上,对于调用者来说,可以通过以下的方式从 Future 对象获得真实的执行结果 Message:
清单 7 是使用该 Active Object 的示例。
MQ_Proxy message_queue; //Optioin 1. Obtain future and block thread until message arrives. Message_Future future = message_queue.get(); Message msg = future.result(); //Handle received message here handle(msg); //2. Obtain a future (does not block the client). Message_Future future = message_queue.get (); //The current thread is not blocked, do something else here... //Evaluate future and block if result is not available. Message msg = future.result (); //Handle received message here handle(msg);
从清单 7 可以看到,MQ_Proxy 对于调用者而言,和一个普通的 C++ 定义的对象并没有区别,并发的实现细节已经被隐藏。
回页首
Java JDK 1.3 引入了 java.util.Timer 和 java.util.TimerTask,提供了对 timer-based 并发任务支持,Timer 和 TimerTask 可以看作是 Active Object 设计模式在 Java 中的实现。不过,在这里我们不打算过多讨论 Timer 及其 TimerTask。由于 Timer 和 TimerTask 的缺陷性,例如 Timer 使用单线程执行 TimerTask 导致的 Task 调度时间的不精确性等问题。从 Java1.5 开始,Java 建议使用 ScheduledThreadPoolExecutor 作为 Timer 的替代。
在这里,我们讨论一下自 Java1.5 引入的 Executor Framework。Java1.5 的 Executor Framework 可以看作是 Active Object 设计模式在 Java 中的体现。不过 Java 的 Executor Framework 极大地简化了我们前面所讨论的 Active Object 所定义的模式。
Java 的 Executor Framework 是一套灵活强大的异步任务执行框架,它提供了标准的方式解耦合任务的提交与任务的执行。Java Executor 框架中的任务指的是实现了 Runnable 或者 Callable 接口的对象。Executor 的示例用法如清单 8 所示:
public class TaskExecutionTcpServer { private static final int NTHREADS = 100; private static final Executor exec = Executors.newFixedThreadPool(NTHREADS); public static void main(String[] args) throws IOException { ServerSocket socket = new ServerSocket(80); while (true) { final Socket connection = socket.accept(); Runnable task = new Runnable() { public void run() { handleRequest(connection); } public void handleRequest(Socket connection) { // Handle the incoming socket connection from //individual client. } }; exec.execute(task); } } }
在示例 8 中,我们创建了一个基于线程池的 Java Executor, 每当新的 TCP 连接进来的时候,我们就分配一个独立的实现了 Runnable 任务来处理该连接,所有这些任务运行在我们创建的有 100 个线程的线程池上。
我们可以从 Active Object 设计模式的角度来审视一下 Java Executor 框架。Java Executor 框架以任务 (Task) 为中心,简化了 Active Object 中的角色分工。可以看到,实现 Runnable 或者 Callable 接口的 Java Executor 任务整合了 Method Request 和 Servant 的角色 , 通过实现 run() 或者 call() 方法实现应用逻辑。Java Executor 框架并没有显式地定义 Proxy 接口,而是直接调用 Executor 提交任务,这里的 Executor 相当于 Active Object 中调度者角色。从调用者的角度来看,这看起来并不像是在调用一个普通对象方法,而是向 Executor 提交了一个任务。所以,在这个层面上说,并发的底层细节已经暴露给了调用者。对于 Java 的开发者来说,如果你不担心这样的底层并发细节直接暴露给调用者,或者说你的应用并不需要像对待普通对象一样对待并发对象,Java 的 Executor 框架是一个很好的选择。相反,如果你希望隐藏这样的并发细节,希望像操纵普通对象一样操纵并发对象,那你就需要如本文上节所描述的那样,遵循 Active Object 设计原则 , 清晰地定义各个角色,实现自己的 Active Object 模式。
总而言之,Java Executor 框架简化了 Active Object 所定义的模式,模糊了 Active Object 中角色的分工,其基于生产者 / 消费者模式,生产者和消费者基于任务相互协作。
回页首
最后,我们讨论一下 Active Object 设计模式的优缺点。
Active Object 给我们的应用带来的好处:
当然,Active Object 也有缺点: