C#中利用代理实现观察者设计模式详解
发布时间:2022-02-25 17:49:36
界面开发中,经常使用观察者设计模式来实现文档/视图模式,当文档内容改变时,作为观察者的用户视图必须相应作出调整以向用户呈现文档的状态。由于语言机制的不同,观察者设计模式在不同的语言中实现方法也不尽相同。
在MFC的文档/视图模式中,每当文档内容改变都需要调用UpdateAllView函数来更新视图,该函数会遍历文档的每一个视图,调用每个视图的更新函数来更新视图,为此文档须登记每一个使用该文档的视图。C#中观察者设计模式的实现也可以采用这种方法,但C#提供的代理(delegate)机制为实现观察者模式提供了更好的方法,该方法和MFC中的方法类似 ,只不过将视图向文档注册这一行为改变为为文档类的代理生成实例而已,下面看具体实现方法。
先做如下假定:
1、文档类为UserData;
2、视图类为View,实际应用中该View可能是一个Form,也可能是一个UserControl,可能有多个视图,但每一个和文档的对应方式都是相同的;
3、主窗体为MainForm;
参与观察者模式的三方分别为:发布者(数据/文档类)、订阅者(视图类)以及主窗体(MainForm),下面分别介绍各方如何实施以配合观察者模式的实现!
发布者:
发布者的任务是定义数据并在数据改变时通知订阅者。通知的实现可以使用普通代理,也可以使用事件,首先在UserData中创建代理和事件,每一个事件在UserData类相应属性改变时触发,看下面的代码:
public delegate void UserNameChangedEventHander(object sender, EventArgs e); //声明代理
public event UserNameChangedEventHander NameChanged; //声明事件
private string m_userName;
public string UserName//定义属性
{
get
{
return m_userName;
}
set
{
if (m_userName != value)
{
m_userName = value;
NameChanged(this, EventArgs.Empty); //触发事件
}
}
}
上述代码首先声明了代理,然后声明了代理对应的事件(事件也算一种特殊的代理),这些代理实例的生成将在视图中进行,然后在属性的set函数中触发事件,该事件将在各个订阅者中得到响应。
订阅者:
订阅者的任务是响应发布者发布的数据改变通知,呈现给用户实时(相对来说)的系统状态。
看下面的代码:
private UserData m_userData = null;
public UserData UserDataObj //定义数据(文档)对象
{
get
{
return m_userData;
}
set
{
m_userData = (UserData)value; //下面一行添加数据对象事件响应函数
m_userData.NameChanged += new UserData.UserNameChangedEventHander(UserNameChanged);
}
}
private void UserNameChanged(object sender,EventArgs e) //定义数据对象事件响应函数
{
this.tbName.Text = m_userData.UserName;//根据数据对象更新内容
this.Invalidate(); //重绘视图
}
上述代码首先在视图类中定义一数据对象属性,并在属性的set函数中添加对数据对象所发布通知的响应。接下来定义了响应数据对象通知的函数,在该函数中更新视图数据并重绘。
主窗体:
主窗体的任务是定义一个相当于全局的数据对象,将其赋予每个订阅该对象的视图,并在需要的时候改变数据对象内容。
看下面的代码:
private UserData m_userData; //发布者
private View m_view;//订阅者
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;
}
在框架窗体类中分别生成发布者和订阅者的实例,然后将发布者实例赋值给订阅者的数据对象属性,由于C#中类的传递默认采用引用传递的方式,因此在赋值过程中并不生成临时对象,MainForm中的m_userData和View中的m_userData所指为同一对象。接下来在主窗体中改变发布者数据,通过C#的代理(delegate)机制,订阅者即能更新自己。
小结
MFC为开发者搭好了框架,尽管作了许多的开发,可能很多人还是不太了解什么是所谓的观察者模式,C#提供了全开放的设计,可能辛苦些,但不再摸不着头脑,条理感觉更清晰些,封装性感觉也比MFC好些!
学习模式注重精髓而非模板,本文为了便于说明假定了三方并对三方功能进行了划分,实际应用并不拘泥于此。如果情况合适将数据(文档)类设计为单件模式也是一种很不错的选择!总之一句话:掌握精髓,尽情发挥!


猜你喜欢
- 1. 前言我们知道,在日常开发中使用的 HashMap 是线程不安全的,而线程安全类 HashTable 和 SynchronizedMap
- 背景众所周知,所有被打开的系统资源,比如流、文件或者Socket连接等,都需要被开发者手动关闭,否则随着程序的不断运行,资源泄露将会累积成重
- 目录1、java有8种基本类型,请问byte、int、long、char、float、double、boolean各占多少个字节?2、在 A
- 本文实例为大家分享了java模拟斗地主发牌的具体代码,供大家参考,具体内容如下1.案例介绍规则:组装54张扑克牌54张牌顺序打乱三个玩家参与
- 文章来源:csdn 作者:chensheng913对于Java语言,最体贴的一项设计就是它并没有打算让人们为了写程序而写程序——人们也需要考
- 常见尺寸单位Android开发中的常用尺寸单位有如下几种:dp (dip)pxptinchsp算不知道确切含义,相信对于以上这几种尺寸单位大
- Android 显示刷新频率android11-release 开发者选项->显示刷新频率packages/apps/Settings
- 前言最近使用QT中的QTextEdit控件,作为实时数据显示的UI,在一次写入超过多少k的时候循环写入则会卡顿,网上也没有什么好的解决方案,
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 今天看到EOE问答里面有这“[Android 界面]NotificationManager 如何使用Bitmap做图标”这样一个问题,在论坛
- 在gitee上创建springcloud仓库 application.yaml(https方式)server: por
- 本文实例讲述了Java设计模式之模板方法模式。分享给大家供大家参考,具体如下:我们在生活中,很多事情都包含特定的一些步骤。如去银行办理业务,
- 前言String可以说是Java中使用最多最频繁、最特殊的类,因为同时也是字面常量,而字面常量包括基本类型、String类型、空类型。一.
- 1. 为什么要进行参数校验在后端进行工作时,需要接收前端传来的数据去数据库查询,但是如果有些数据过于离谱,我们就可以直接把它pass掉,不让
- mybatis foreach嵌套if标签代码实现:Mapper.java文件List<Map<String, Object&g
- 本文介绍了浅谈Java的两种多线程实现方式,分享给大家。具有如下:一、创建多线程的两种方式Java中,有两种方式可以创建多线程:1 通过继承
- 获取本机所有IP地址:这些地址是包含所有网卡(虚拟网卡)的ipv4和ipv6地址。 string name = Dns.GetHostNam
- Android游戏开发中主要的类除了控制类就是显示类,
- 前言最近项目中又一次需要集成友盟的三方登录与分享,之前没有记录过,所以这次来写一下...准备工作1.注册友盟账号创建应用,获取key:申请地
- 基本步骤三数取中在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。在此我们采用三数取中法,也就是取左端、中间