界面开发中,经常使用观察者设计模式来实现文档/视图模式,当文档内容改变时,作为观察者的用户视图必须相应作出调整以向用户呈现文档的状态。由于语言机制的不同,观察者设计模式在不同的语言中实现方法也不尽相同。
在MFC的文档/视图模式中,每当文档内容改变都需要调用UpdateAllView函数来更新视图,该函数会遍历文档的每一个视图,调用每个视图的更新函数来更新视图,为此文档须登记每一个使用该文档的视图。C#中观察者设计模式的实现也可以采用这种方法,但C#提供的代理(delegate)机制为实现观察者模式提供了更好的方法,该方法和MFC中的方法类似 ,只不过将视图向文档注册这一行为改变为为文档类的代理生成实例而已,下面看具体实现方法。
先做如下假定:
1、文档类为UserData;
2、视图类为View,实际应用中该View可能是一个Form,也可能是一个UserControl,可能有多个视图,但每一个和文档的对应方式都是相同的;
3、主窗体为MainForm;
参与观察者模式的三方分别为:发布者(数据/文档类)、订阅者(视图类)以及主窗体(MainForm),下面分别介绍各方如何实施以配合观察者模式的实现!
发布者:
发布者的任务是定义数据并在数据改变时通知订阅者。通知的实现可以使用普通代理,也可以使用事件,首先在UserData中创建代理和事件,每一个事件在UserData类相应属性改变时触发,看下面的代码:
订阅者:
订阅者的任务是响应发布者发布的数据改变通知,呈现给用户实时(相对来说)的系统状态。
看下面的代码:
主窗体:
主窗体的任务是定义一个相当于全局的数据对象,将其赋予每个订阅该对象的视图,并在需要的时候改变数据对象内容。
看下面的代码:
private void MainForm_Load(object sender, EventArgs e)
{
m_userData = new UserData(); //生成实例
m_view = new View();
m_view.UserDataObj = m_userData; //为订阅者指定发布者
m_view.Show(); //显示
m_userData.UserName = "ZPY"; //改变发布者数据
m_view.TopMost = true;
}
小结
MFC为开发者搭好了框架,尽管作了许多的开发,可能很多人还是不太了解什么是所谓的观察者模式,C#提供了全开放的设计,可能辛苦些,但不再摸不着头脑,条理感觉更清晰些,封装性感觉也比MFC好些!
学习模式注重精髓而非模板,本文为了便于说明假定了三方并对三方功能进行了划分,实际应用并不拘泥于此。如果情况合适将数据(文档)类设计为单件模式也是一种很不错的选择!总之一句话:掌握精髓,尽情发挥!
新闻热点
疑难解答