我在學(xué)習的過(guò)程中也理解到以上原則的重要性和指導性,在我學(xué)習模塊狀態(tài)和Event Broker的過(guò)程中,也將上述部分原則做了特意的應用。那么我們還是通過(guò)一個(gè)實(shí)例來(lái)學(xué)習Event Broker和這些原則。
一、文中有關(guān)術(shù)語(yǔ)
下面這些術(shù)語(yǔ)是CAB中常用到的,以下的解釋僅是我個(gè)人的理解,不敢保證完全準確,園子里的朋友請指教。
Event Broker:事件代理,通過(guò)事件源和訂閱事件源來(lái)達成對象之間的協(xié)作。
Event Publisher: 事件發(fā)布者,在CAB里是一個(gè)用屬性EventPublication修飾的事件對象,提供特定的URL給Event Subscriber訂閱。
Event Subscriber: 事件訂閱者,在CAB里是一個(gè)用屬性EventSubscription修飾的方法,根據修飾提供的URL自動(dòng)尋找事件發(fā)布者。Publisher和Subscriber之間由主題(由URL決定),消息(特定的 EventArgs),事件域(來(lái)確定是全局事件還是局部事件)來(lái)達成一致。其實(shí)這也是觀(guān)察者模式的具體實(shí)現。
WorkItem:代表一個(gè)用例,也可以看成是某個(gè)業(yè)務(wù)完成的過(guò)程,它包含在WorkSpace中,服務(wù)于Service Agents(服務(wù)代理),并且加載其狀態(tài)。創(chuàng )建其他組件或者視圖,CAB來(lái)創(chuàng )建controller.組件共享WorkItem的狀態(tài),并且可以通過(guò)狀態(tài)來(lái)控制用例的生命周期。
WorkItem State:狀態(tài),實(shí)際上是把業(yè)務(wù)對象或者業(yè)務(wù)對象的屬性,通過(guò)WorkItem State共享出來(lái),方便其他業(yè)務(wù)對象或者視圖訪(fǎng)問(wèn)。
二、體驗Event Broker應用
講了這么多有關(guān)Event Broker的理論和概念了,我們還是通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)體驗Event Broker這種實(shí)現模式的優(yōu)越性吧。
1.應用場(chǎng)景
平時(shí)我們在開(kāi)發(fā)過(guò)程中碰到最多的例子大概就是,一個(gè)業(yè)務(wù)對象數據集要通過(guò)dataGrip,ListBox甚至Chart控件等將其表現出來(lái)了。今天,我在學(xué)習筆記里也以這個(gè)例子來(lái)闡述Event Broker,在開(kāi)發(fā)中帶來(lái)的好處。
場(chǎng)景是這樣的:某人事信息管理軟件要求輸入人員的性別和姓名,并且能將輸入的人員在通過(guò)表格和列表框的形式表現出來(lái),同時(shí)錄入人員的男女比例要能適時(shí)的通過(guò)餅圖顯示。
2.分析場(chǎng)景,確定開(kāi)發(fā)模式
a.需求中涉及到的唯一業(yè)務(wù)對象是人員,具有性別和姓名兩個(gè)屬性。為了簡(jiǎn)單起見(jiàn)我們可以建立數據集來(lái)代替該對象。
b.需求要求能輸入姓名、性別,我們可以用文本框和下拉框來(lái)完成信息采集。
c.需求要求人員信息,通過(guò)表格,ListBox和餅圖來(lái)顯示,我們可以在VS2005中用DataGrid、ListBox、ReportView來(lái)實(shí)現此項需求。
d.由于業(yè)務(wù)對象單一,而信息表現卻又多個(gè),適合用觀(guān)察者模式進(jìn)行開(kāi)發(fā)。我們便采用CAB中的Event Broker作為重要的實(shí)現手段。
3.建立應用程序
第一步:新建項目
啟動(dòng)VS2005,新建Windows Application,添加以下引用:
Microsoft.Practices.CompositeUI
Microsoft.Practices.CompositeUI.WinForms
Microsoft.Practices.ObjectBuiler
Microsoft.Practices.CompositeUI.Utility
Microsoft.Practices.CompositeUI.WinForms
第二步:建立數據集
右擊項目文件夾,添加新項,選擇數據集,建立用戶(hù)信息數據集(沒(méi)有通過(guò)代碼創(chuàng )建,主要是為了設計報表方便)。為數據集添加DataTable1的表,為DataTable1添加列Sex和Name。
第三步:繪制界面
在VS2005默認生成的Form1上建立餅圖、DataGrid、ListBox和相關(guān)相關(guān)控件,具體操作我在此略過(guò),最終效果如下圖:
為了讓程序能使用CAB,我們必須修改程序的入口類(lèi)Program.cs。最終修改結果如下:






































需要注意的是:為了能使用WorkItem的State,在Shell創(chuàng )建之前必須給共享的狀態(tài)賦初值,否則在訪(fǎng)問(wèn)該狀態(tài)時(shí)將出現狀態(tài)沒(méi)有創(chuàng )建實(shí)例的運行時(shí)錯誤。本例中就是加入以下代碼:







第五步:建立controller
建立controller負責用戶(hù)信息添加,建立事件源。添加類(lèi)文件,命名為Form1Controller,將該類(lèi)從controller繼承。如下代碼所示:



















在controller中公布一個(gè)事件發(fā)布者,通過(guò)"topic://TestReport/DataRowAdded"來(lái)標識Publisher,默認的事件域為全局。也可以通過(guò)PublicationScope枚舉來(lái)設置事件的作用域。事件作用域有以下三種:
PublicationScope.WorkItem :僅作用于引發(fā)當前發(fā)布的WorkItem實(shí)例
PublicationScope.Global:作用于引發(fā)當前發(fā)布的WorkItem所有實(shí)例
PublicationScope.Descendants:僅作用于引發(fā)當前發(fā)布的WorkItem實(shí)例,以及該WorkItem的任何級別的子WorkItem實(shí)例。
本例通過(guò)以下代碼發(fā)布事件:
[EventPublication("topic://TestReport/DataRowAdded")]
public event EventHandler<DictionaryEventArgs> DataRowAdded;
controller中主要來(lái)實(shí)現業(yè)務(wù)邏輯,于是我們需要添加一個(gè)方法AddNewRow(int sex, string name),用來(lái)實(shí)現人員信息的添加,代碼如下:
























大家請注意下面代碼,其實(shí)是定義了一個(gè)DictionaryEventArgs參數,并且將當前添加的行對象作為該參數的值。當DataTable1中行添加后,我們引發(fā)事件DataRowAdded(this, args)。 此時(shí),事件源被觸發(fā)了,訂閱者就可以接收到該事件廣播了。
DictionaryEventArgs args = new DictionaryEventArgs();
args.Data["dataRow"] = myRow;
DataRowAdded(this, args);
到此,我們已經(jīng)完成了事件源的創(chuàng )建和發(fā)布,為了達到演示的效果,我們還需要實(shí)現共享WorkItem State來(lái)廣播事件。如以下代碼:


































為添加按鈕加入代碼,實(shí)現添加一個(gè)人員信息:












還有為了讓Grid和report view能夠同步顯示人員信息,我們需要訂閱由topic://TestReport/DataRowAdded標示的事件:









這樣每添加一個(gè)人員,Grid和Reoport View就能適時(shí)更新自身表現了,這就是Event Broker的實(shí)現方式,簡(jiǎn)單并且簡(jiǎn)潔。前面我們還提到了通過(guò)共享狀態(tài)來(lái)實(shí)現視圖和業(yè)務(wù)對象的關(guān)聯(lián),在本例中也提供實(shí)現。
首先,在FormLoad事件中訂閱StateChanged事件:







然后,通過(guò)代碼更新List狀態(tài):








好了,到此我們的例程已經(jīng)大功告成,最終的運行效果如下圖:

聯(lián)系客服