用過Flex的人都知道有個基于MVC的puremvc框架,這個框架因為小巧簡單很受歡迎。
PureMVC框架的目標很明確,即把程序分為低耦合的三層:Model、View和Controller。降低模塊間的耦合性,各模塊如何結(jié)合在一起工作對于創(chuàng)建易擴展,易維護的應用程序是非常重要的。在PureMVC實現(xiàn)的經(jīng)典MVC元設(shè)計模式中,這三部分由三個單例模式類管理,分別是Model、View和Controller。三者合稱為核心層或核心角色。PureMVC中還有另外一個單例模式類—Facade,Facade提供了與核心層通信的唯一接口,以簡化開發(fā)復雜度。
這里先放這個PureMVC這張著名的圖:
這里就簡單介紹了PureMVC一些簡單知識,詳情可以到puremvc.org 查看。這里我們不大去講他的原理,只是通過一個例子來簡單說下PureMVC的機制如何來運行一個Swing項目。
回到本文重點,因為PureMVC的小巧簡單,所以作者也順帶著把他翻譯成了C#,ColdFusion,Haxe,Java ,JavaScript ,Objective C ,PHP ,Python ,Ruby,那現(xiàn)在我們就以基于Java來做個Demo來講講PureMVC的運行原理。這個例子是一個小型系統(tǒng)的簡單實現(xiàn),首先彈出一個對話框,輸入用戶名和密碼成功后進入系統(tǒng),進行相關(guān)操作,所有動作都是利用PureMvc的機制發(fā)送消息,做到MVC的三層Model、View和Controller分離。
由上圖知道我們,程序的入口是Facade進去的,這個類是一個單類,一般我們都會繼承這個類重寫Facade,在程序啟動的時候用他我們來注冊命令等相關(guān)信息。所有的信息發(fā)送通過他來發(fā)送。在使用的時候我們一般繼承他的initializeController方法用于加載初始化的一些Command,比如點擊菜單打開一個界面的信息,都是在這里預先注冊,如代碼:

public class ApplicationFacade extends Facade
{
private static ApplicationFacade instance= null;

protected ApplicationFacade()
{
super();
}

public static ApplicationFacade getInstance()
{
if(instance == null)
instance = new ApplicationFacade();
return instance;
}
@Override

protected void initializeController()
{
super.initializeController();

/** *//**
* 注冊初始化的Command
* **/
registerCommand(ApplicationConstants.INITSTAR, InitStartCommand.class);
}

/** *//**
* 系統(tǒng)初始化完畢后調(diào)用這個發(fā)送INITSTAR,監(jiān)聽這個命令的Command
* 就會加載所有的Command
* **/

public void startup()
{
sendNotification(ApplicationConstants.INITSTAR, null, null);
}
}

注冊完了預加載的Command的命令后,我們的系統(tǒng)初始化完畢后點擊一個按鈕需要打開一個頁面,這個事件就會發(fā)送出一個消息,至于發(fā)給誰他不管,這其中如果有注冊了的Command監(jiān)聽了這個發(fā)送的消息,那么他就會做進一步的操作,如圖所示:
當這個消息給send出來后,當有Command接受到這個消息后就會做他感興趣的事。所有的發(fā)送消息的方法都是通過sendNotification這個方法發(fā)送的這個方法是這樣定義的:
public void sendNotification(String arg0, Object arg1, String arg2);
arg0:就是發(fā)送的消息,常量字符串,如上面代碼的INITSTAR,一般都會放在專門存放常量表中。
arg1:可以存放任何東西,比如要傳一些值過去,就可以放在這里面,比如在接下來的登入成功后就會把用戶的信息,發(fā)送到主界面中去,就可以這樣定義:
Map data = new HashMap();
data.put("userName", name);
facade.sendNotification(ApplicationConstants.LOGIN_SUCESS, data, null);
arg2:一般可以為空,或其他什么標識等。
當發(fā)送完一個命令對應的Command監(jiān)聽到,他會在一個execute方法中做出相應的操作如:
@Override

public void execute(INotification noti)
{
super.execute(noti);

if(ApplicationConstants.SHOW_FRAME_01.equals(noti.getName()))
{
//打開InternalFrame01
doShow01();

}else if(ApplicationConstants.SHOW_FRAME_02.equals(noti.getName()))
{
//打開InternalFrame02
doShow02();
}
}
當偵聽到一個命令比如執(zhí)行doShow01打開一個頁面,這個頁面如果需要和其他頁面有消息接發(fā)的時候,那么在這里就主要注冊一個Mediator,這個Mediator要重寫兩個方法,

public String[] listNotificationInterests( )
{

return new String[]
{
ApplicationConstants.LOGIN_SUCESS,
ApplicationConstants.CREATE_FRAME,
ApplicationConstants.EXIT,
ApplicationConstants.SEND_TO_FRAME_01,
ApplicationConstants.SEND_TO_FRAME_02
};
}

這個listNotificationInterests方法注冊我要監(jiān)聽什么消息。
還一個handleNotification方法如:

public void handleNotification(INotification noti)
{
// TODO Auto-generated method stub
super.handleNotification(noti);
if(ApplicationConstants.LOGIN_SUCESS.equals(noti.getName()))
doLoginSucess(noti);
}
}

這個handleNotification方法,是執(zhí)行監(jiān)聽到那些消息后執(zhí)行具體的動作。也就說一個命令發(fā)送會有Command和Mediator都接受到。大致這個框架就是這樣處理的。
通過上面的介紹,我們就結(jié)合一個例子來簡單實現(xiàn)下,這個例子大意是程序啟動后出現(xiàn)一個登入框,任意輸入用戶名和密碼點擊登入,就會發(fā)送一個消息LOGIN_SUCESS,那么這主界面對應的MainFrameMediator監(jiān)聽到這消息就會顯示出程序的主界面,這個主界面的效果圖為
其中點擊菜單欄的Frame01就會出現(xiàn)標題欄為:Kissjava的一個JinternalFrame,點擊Frame02會出現(xiàn)一個標題欄為Rocky Jiang的JinternalFrame,在標題欄為Kissjava頁面中的輸入框中輸入一些信息點擊發(fā)送一個SEND_TO_FRAME_02消息,這個消息會有對應的標題欄為Rocky Jiang的頁面對應的Mediator和主界面的Mediator監(jiān)聽到。所以在各自對應的日志輸入框中都會出現(xiàn)對應的消息。剛興趣的人可以在后面提供的鏈接下載代碼自己運行看看。代碼結(jié)構(gòu)簡單。
使用PureMVC機制,可以很好的實現(xiàn)MVC三層很好的分離,在本例子中由于例子簡單就沒涉及到Model層,這個Model它只負責有數(shù)據(jù)更改的時候發(fā)送消息,但不會接受消息,這樣才能做到更好的分離。
當然使用這個還涉及到一些不用的對象銷毀等問題,這里就不詳說了。
源代碼下載:DOWNLOAD