使用C# (.NET Core) 实现观察者模式 (Observer Pattern) 并介绍 delegate 和 event
观察者模式
这里面综合了几本书的资料.
需求
有这么个项目:
需求是这样的:
一个气象站, 有三个传感器(温度, 湿度, 气压), 有一个WeatherData对象, 它能从气象站获得这三个数据. 还有三种设备, 可以按要求展示气象站的最新数据.
WeatherData的结构如下:
有3个get方法, 分别获取最新的气温, 湿度和气压. 还有一个measurementsChanged()方法, 当任一传感器有变化的时候, 这个方法都会被调用.
总结一下项目的需求:
- WeatherData类有三个get方法可以获取温度, 湿度和气压
- 如果任何一个数据发生变化, 那么measureChanged()方法就会被调用
- 我们需要实现这三种显示设备:
- 当前天气
- 数据统计
- 天气预测
- 系统必须可以扩展, 其他开发者可以创建自定义展示设备.
初版代码
这个地方有个"错误", xxxDisplay都是具体的实现, 而编程规则要求是应该对接口编程而不是对实现编程.
那么什么是观察者模式?
举一个例子:
- 报社发行报纸
- 你订阅报纸, 一旦有新一期的报纸发行, 新报纸就会送到你家里, 只要你一直订阅, 你就一直会收到新报纸
- 你不再订阅报纸的时候, 就收不到以后的新报纸了
- 报社运营的时候, 一直会有人去订阅或者取消订阅报纸.
发布者 + 订阅者 = 观察者模式
Publishers + Subscribers = Observer Pattern
在观察者模式里, 我们把报社叫做被观察对象(Subject), 把订阅者叫做观察者(Observers)
观察者模式是这样操作的:
观察者模式的定义就是:
一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知。
类图如下:
谈一下松耦合
当两个对象是松耦合的时候, 他们可以进行交互, 但是却几乎不了解对方.
观察者模式下的被观察者(Subject)和观察者(Observers)就是松耦合设计的对象. 这是因为:
- 被观察者(Subject)只知道观察者实现了某个接口
- 可以随时添加观察者
- 添加新类型观察者的时候不需要修改被观察者
- 可以复用观察者或者被观察者
- 如果被观察者或观察者发生变化了, 那么这些变化不会影响到对方.
一个设计原则:
交互的对象之间应尽量设计成松耦合的. Strive for loosely coupled designs between objects that interact.
松耦合设计可以让我们设计出这样的系统: 因为对象之间的相互依存减小了, 所以系统可以轻松处理变化.
重新设计:
代码:
OK, 上面是书中的内容, C#7.0里面对观察者模式是怎么实现的呢?
先只谈下面这个:
Event
谈到Event, 就得把delegate先细说一下
Delegate 委托
一个委托类型定义了某种类型的方法(方法的返回类型和参数类型), 然后这个委托的实例可以调用这些方法.
例如:
delegate int Transformer (int x);
这个委托就和返回类型是int, 参数是一个int的方法兼容.
例如:
static int Square (int x) { return x * x }; // 或 static int Square (int x) => x * x;
把一个方法赋值给委托变量的时候就创建了一个委托的实例:
Transformer t = Square;
然后就可以像方法一样进行调用:
int answer = t(3); // 9
所以说一个委托的实例就是调用者的委托: 调用者调用委托, 然后委托调用目标方法, 这样就把调用者和目标方法解耦了.
其中:
Transformer t = Square; // 是下面的简写 Transformer t = new Transformer(Square);
t(3) // 是下面的简写 t.Invoke(3)
多播委托
一个委托实例可以引用多个目标方法. 使用+=操作符.
SomeDelegate d = Method1; d += Method2; // 第二行相当于: d = d + Method2;
调用d的时候就会调用Method1和Method2两个方法.
委托方法的调用顺序和它们被添加的顺序是一样的.
使用-=操作符来移除目标方法:
d -= Method1;
这时调用d后只会执行Method2了.
注意: 委托是不可变的 +=/-=实际上是创建了新的委托.
多播委托返回类型
如果多播委托有返回值(非void), 那么调用者只会获得最后一个被调用方法的返回值.
委托也可以使用泛型:
public delegate T Transformer<T> (T arg);
Func 和 Action
记住Func有返回值, Action没有就行.
Event
使用委托的时候, 通常会有两个角色出现: 广播者(被观察者)和订阅者(观察者) [观察者模式]
广播者包含一个委托field, 广播者决定何时广播, 它通过调用委托进行广播.
订阅者就是方法的目标接收者.订阅者可以决定何时开始和结束监听, 是通过在广播者的委托上使用+=和-=操作符来实现的.
订阅者之间互相不了解, 不干扰.
event就是为上述模型所存在的, 它只把上述模型所必须的功能从委托里暴露出来. 它的主要目的就是防止订阅者之间相互干扰.
最简单声明event的方法就是在委托成员前面加上event关键字:
public delegate void SomeChangedHandler(decimal x); public class Broadcaster { public event SomeChangedHandler handler; }
在Broadcaster类里面的代码, 可以把handler作为委托一样来用.
在Broadcaster类外边, 只能对这个event执行+=和-=操作.
Event 模式/ 观察者模式
这种模式在.net core里首先需要EventArgs.
EventArgs是一个基类, 它可以为event传递信息.
可以创造它的子类来传递自定义参数:
public class FallsIllEventArgs : EventArgs { public readonly string Address; public FallsIllEventArgs(string address) { this.Address = address; } }
然后就需要给这个event定义一个委托了, 这有三条规则:
- 返回类型必须是void
- 需要有两个参数, 第一个是object, 第二个是EventArgs的子类. 第一个参数代表着广播者, 第二个参数包含额外的需要传递的信息.
- 名称必须以EventHandler结束.
.net core定义了System.EventHandler<>, 它满足这些要求.
public event EventHandler<FallsIllEventArgs> FallsIll;
最后, 需要写一个 protected virtual 方法可以触发event. 方法的名称必须和event匹配: 以On开头, 接受EventArgs类型的参数:
public void OnFallsIll() { FallsIll?.Invoke(this, new FallsIllEventArgs("China Beijing")); }
注意: 预定义的非泛型的EventHandler委托可以在没有数据需要传输的时候使用, 调用的时候可以使用EventArgs.Empty来避免不必要的初始化EventArgs.
用.net core 实现观察者模式的代码:
Person.cs
using System; namespace ObserverPattern { public class Person { public event EventHandler<FallsIllEventArgs> FallsIll; public void OnFallsIll() { FallsIll?.Invoke(this, new FallsIllEventArgs("China Beijing")); } } }
using System; namespace ObserverPattern { public class FallsIllEventArgs : EventArgs { public readonly string Address; public FallsIllEventArgs(string address) { this.Address = address; } } }
using System; namespace ObserverPattern { class Program { static void Main(string[] args) { var person = new Person(); person.FallsIll += OnFallsIll; person.OnFallsIll(); person.FallsIll -= OnFallsIll; } private static void OnFallsIll(object sender, FallsIllEventArgs eventArgs) { Console.WriteLine($"A doctor has been called to {eventArgs.Address}"); } } }
下面是我的关于ASP.NET Core Web API相关技术的公众号--草根专栏:

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
泛型就这么简单
前言 从今天开始进入Java基础的复习,可能一个星期会有一篇的<十道简单算法>,我写博文的未必都是正确的~如果有写错的地方请大家多多包涵并指正~ 今天要复习的是泛型,泛型在Java中也是个很重要的知识点,本文主要讲解基础的概念,并不是高深的知识,如果基础好的同学可以当复习看看~ 一、什么是泛型? Java泛型设计原则:只要在编译时期没有出现警告,那么运行时期就不会出现ClassCastException异常. 泛型:把类型明确的工作推迟到创建对象或调用方法的时候才去明确的特殊的类型 参数化类型: 把类型当作是参数一样传递 <数据类型> 只能是引用类型 相关术语: ArrayList<E>中的E称为类型参数变量 ArrayList<Integer>中的Integer称为实际类型参数 整个称为ArrayList<E>泛型类型 整个ArrayList<Integer>称为参数化的类型ParameterizedType 二、为什么需要泛型 早期Java是使用Object来代表任意类型的,但是向下转型有强转的问题,这样程序就...
- 下一篇
虚拟机字节码执行引擎
一、概述 物理机的执行引擎:直接建立在处理器、硬件、指令集和操作系统层面 虚拟机的执行引擎:由自己实现,可以自行制定指令集与执行引擎的结构体系,并且能够执行不被硬件直接支持的指令集格式。 java虚拟机的执行引擎:输入字节码文件,处理过程是字节码解析的等效过程,输出是执行结果。 二、运行时栈帧结构 栈帧:用于支持虚拟机进行方法调用和方法执行的数据结构,是虚拟机运行时数据区中的虚拟机栈的栈元素。 栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。每个方法从调用开始至执行完成的过程都对应一个栈帧在虚拟机栈里从入栈到出栈的过程。 在编译程序代码时,栈帧需要多大的局部变量表、多深的操作数栈都已经完全确定,并写入方法表Code属性中,因此一个栈帧需要分配多少内存,不会受程序运行期数据的影响。 对于执行引擎来说,在活动线程中,只有位于栈顶的栈帧才是有效的,称为当前栈帧(current stack frame),与这个 栈帧相关联的方法称为当前方法(current method)。执行引擎运行的所有字节码指令都只针对当前栈帧进行操作。 1、局部变量表 局部变量表(local var...
相关文章
文章评论
共有0条评论来说两句吧...