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






































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







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



















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
























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





















??? 我們注意到[State("dataset")]這行代碼,它是用來表示W(wǎng)orkItem的屬性CtlDataSet,將通過[State("dataset")]共享出去,同時當(dāng)CtlDataSet改變時,通過代碼State.RaiseStateChanged("dataset", myRow),來引發(fā)狀態(tài)改變事件,其他地方就可以得到該事件的委托。
第六步:整合界面和controller
??? 我們回到Form1.cs編輯代碼。為了讓界面和controller和界面結(jié)合,我們將controller作為界面對象的一個屬性,用以下代碼實現(xiàn):













?
?? 為添加按鈕加入代碼,實現(xiàn)添加一個人員信息:












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









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







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








好了,到此我們的例程已經(jīng)大功告成,最終的運行效果如下圖:
本文相關(guān)代碼通過此連接下載:
/Files/hyphappy/TestReport.rar
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
