一、需求分析 在JavaScript里采用OO編程思想后,和其它語言一樣(Java, C/C++等)通常都會面臨對象間通信的問題。我們知道一般意義上的對象間通信,就是對象方法的調(diào)用,比如對象A要驅(qū)動對象B完成某一任務(wù),那么對象A會調(diào)用對象B的一個方法,并傳入相應(yīng)的參數(shù)。如圖:
不過這種通信方式有一個前提,就是對象A可以直接訪問到對象B,也就是說對象A和對象B之間是緊密耦合的。然而,在有些時候(也許是很多時候),對象A也許不能直接訪問對象B,甚至于對象A可能都不知道對象B的存在。那么這種情況下如果對象A還要驅(qū)動對象B完成某一任務(wù),就只有借助所謂的Message系統(tǒng)來傳遞信息了。如圖:
當(dāng)然這種通信方式也有一個前提,就是對象A,對象B都需要知道Messenger,不過這不是問題,因為Messenger通常都是作為系統(tǒng)服務(wù)提供。 二、 用例設(shè)計
在JavaScript里如何實現(xiàn)Messenger服務(wù)呢,方法很多,這里提供一種常見的,基于訂閱/發(fā)布模式(subscribe/publish)的簡單實現(xiàn)。所謂訂閱/發(fā)布模式很好理解,就像你從郵局訂閱某種期刊,當(dāng)期刊的發(fā)行者有新的期刊時,他會把期刊交給郵局,而郵局負(fù)責(zé)把期刊投送到你手里。可以用下面這個用例圖來描述這個郵局:
用JavaScript如何描述這個郵局呢,既然是OO編程,我們自然還是用對象來定義這個郵局,由于比較偏向Java的風(fēng)格,所以就沿用了Java的類名字java.awt.EventQueue(其實,我本人從未用過這個類,但大概能猜出它的含義),我們?yōu)檫@個郵局的類定義如下:
- /**
-
* Define the EventQueue class based on subscribe/publish mode
-
*/
-
js.awt.EventQueue = function(){
-
/**
-
* Subscribe a message from EventQueue
-
*
-
* @param msgId, what
-
* @param listener, who
-
* @param handler, where
-
*/
-
this.register = function(msgId, listener, handler){
-
// TODO: Register to DB
-
-
};
-
-
/**
-
* Schedule system use this method to dispatch a message to
-
* listeners
-
*
-
* @param msg
-
*/
-
this.dispatch = function(msg){
-
// TODO: Dispatch message to listeners
-
-
};
-
-
/**
-
* Message sender use this method to post a message and
-
* return immediately
-
*
-
* @param msg
-
*/
-
this.post = function(msg){
-
// TODO: Put this message to the message queue.
-
-
};
-
-
/**
-
* Message sender use this method to send a message and
-
* blocked till to all receivers process this message completely
-
*
-
* @param msg
-
*/
-
this.send = function(msg){
-
// TODO: Deliver this message to receiver directly
-
-
};
-
-
}.$extend(ls.lang.Object);
上面的代碼中, register 方法對應(yīng) Subscribe 用例, dispatch 方法對應(yīng) Deliver 用例, post/send 方法對應(yīng) Publish 用例,這里還需要對這個“郵局”的功能再注釋一下:
- Subscribe 用例, 對應(yīng) register 方法,用于消息接收者向“郵局”注冊自己的 what, who, where 三要素。
- what 可以理解為消息的種類標(biāo)示;
- who 在OO編程中自然是指消息接收者對象本身;
- where 是指消息接收者使用什么方法來處理消息。
- Publish 用例, 有兩個方法 post 和 send。
- post 方法用于異步消息的發(fā)送,也就是消息的發(fā)送者在 post 消息后, post 方法會立即返 回,而消息交由調(diào)度系統(tǒng)發(fā)送。
- send 方法用于同步消息的發(fā)送, 即消息的發(fā)送者在 send 消息后, send 方法將會被阻塞直到消息的接收者處理完消息。
- Deliver 用例, 對應(yīng) dispatch 方法,這個方法由調(diào)度系統(tǒng)調(diào)用。同時當(dāng)使用 send 方法發(fā)送消息時,消息不通過調(diào)度系統(tǒng)發(fā)送。
- 不限制任何對象既作為消息的發(fā)布者又作為消息的接收者。
- 不限制對象發(fā)送何種消息,任何對象都可以發(fā)送其業(yè)務(wù)規(guī)定的多種type的消息,一種消息就是一種type的 what。
- 不限制對象接收何種消息,任何對象都可以接收其業(yè)務(wù)規(guī)定的多種type的消息,一種消息就對應(yīng)一個 where 來處理這個消息。
未完待續(xù)... |