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好些!
学习模式注重精髓而非模板,本文为了便于说明假定了三方并对三方功能进行了划分,实际应用并不拘泥于此。如果情况合适将数据(文档)类设计为单件模式也是一种很不错的选择!总之一句话:掌握精髓,尽情发挥!
猜你喜欢
- 一、SpringBoot 指定配置文件路径:在 SpringBoot 中,可以将配置文件放在 jar 包外面,这样可以方便地修改配置而不需要
- 在之前项目中有用到关于获取手机联系人的部分,闲置就想和大家分享一下,话不多说,上代码:java部分:package com.example.
- 前言通过深入分析Spring源码,我们知道Spring框架包括大致六大模块, 如Web模块,数据库访问技术模块,面向切面模块,基础设施模块,
- 总结并复现了一下Unsafe在安全领域的一些应用0 前言unsafe里面有很多好用的方法,比如allocateInstance可以直接创建实
- 前言每次update Maven Project 的时候,看着进度条寸步难行,心里憋得十分难受,明显阻碍我学习的热情。 maven仓库默认在
- LeetCode54. 螺旋矩阵 java实现题目难度 中给定一个包含 m x n 个元素的矩阵(m 行, n 列),请按照顺时针螺旋顺序,
- 数据层测试事务回滚pom.xml导入对应的一些坐标,mysql,Mp,等<dependency> &
- 一、什么是ASMASM是一个java字节码操纵框架,它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 class 文件,
- 一,栈1,概念在我们软件应用 ,栈这种后进先出数据结构的应用是非常普遍的。比如你用浏 览器上网时不管什么浏览器都有 个"后退&qu
- JVM运行原理首先从“.java”代码文件,编译成“.class”字节码文件,然后类加载器将“.class”字节码文件中的类给加载带JVM中
- 目前知道的情况被调用的C/C++函数只能是全局函数 不能调用类中的成员方法被调用的C函数必须使用extern “C“包含,保证采用的导出函数
- 多数据源创建数据库CREATE DATABASE mybatis_plus_1;USE mybatis_plus_1;CREATE TABL
- 如下所示:public class 字符串常用操作 { public static void main(String[] arg
- 本文实例为大家分享了JavaFX实现简单日历效果的具体代码,供大家参考,具体内容如下1.先看效果:2.代码:1)ClockEdge.java
- 本文实例讲述了java实现的五子棋游戏代码,分享给大家供大家参考,具体代码如下package gyb.exam;import java.aw
- 在开始之前,我先卖个关子提一个问题:假设我们有一个Movie类,这个类有三个成员变量分别是starred(是否收藏), title(电影名称
- 1.springboot 2.0 默认连接池就是Hikari了,所以引用parents后不用专门加依赖2.贴我自己的配置(时间单位都是毫秒)
- 前言此文适合了解了es相关概念以及基础知识的同学阅读elasticsearch简介Elasticsearch是一个基于Lucene的搜索服务
- 摘要:Java8通过Function获取字段名,解决硬编码,效果类似于mybatis-plus的LambdaQueryWrapper。本文总
- Java未被捕获的异常在你学习在程序中处理异常之前,看一看如果你不处理它们会有什么情况发生是很有好处的。下面的小程序包括一个故意导致被零除错