一、 观察者模式
目的
概述
原理
二、 C#中的观察者模式
概述
模型与观察者基类
优点
三、 事例
题目:猫大叫,两只老鼠开始逃跑,主人醒来,宝宝也醒来了并且哭了起来
解决方案:
1. 建立模型(目标基类)
2. 建立观察者基类(单行为,多行为)
3. 建立具体目标
4. 建立具体观察者
5. 运行测试
一、 观察者模式
目的
我们都知道解决一个问题有N种解决方式,但在面向对象的设计中如何能做到“高内聚,低耦合”,设计可重用的对象才是我们追求的。在设计过程中,我们经常会接触到一种情况:一个对象的行为引发其它多个对象相应的行为。这时我们便可以通过观察者模式的设计思想来设计对象模型。
概述
观察者模式(Observer Pattern)是设计模式中行为模式的一种,它解决了上述具有一对多依赖关系的对象的重用问题。此模式的参与者分为两大类,一类是被观察的目标,另一类是观察该目标的观察者们。正因为该模式是基于“一对多”的关系,所以该模式一般是应用于由一个目标对象和N个观察者对象组成(当然也可以扩展为有多个目标对象,但我们现在只讨论前者)的场合。当目标对象的状态发生改变或做出某种行为时,正在观察该目标对象的观察者们将自动地、连锁地作出相应的响应行为。
原理
我们可以把观察目标理解为主动方、发布方、主体等;把观察者理解为被动方、订阅方、观察器等。目标是整个行为链的源头,其它观察者都依赖于它的变化而作出响应。为了实现低耦合,我们不能使用“直接调用”的方式而需要利用“订阅(清单)-通知”的机制去完成设计。通俗地说就是观察者向目标“订阅”它的改变,而目标发生改变后就“通知”所有已经“订阅”了它的改变的观察者,从而执行“订阅”的内容。这种机制的好处在于降低耦合度,分工明确,目标只负责在自身状态发生改变或做出某种行为时向自身的订阅清单发出“通知”,而不是直接调用观察者的行为(方法);观察者只负责向目标“订阅”它的变化,以及定义自身在收到目标“通知”后所需要做出的具体行为(也就是订阅的内容)。就像我们向出版社订阅报刊一样,出版社有新一期报刊发行时并不是直接跟每位订阅者联系,而是“通知”订阅者名单按顺序给每位订阅者发送所订报刊。
二、 C#中的观察者模式
概述
每种编程架构及程序语言,对观察者模式都有不通的具体实现。在.NET框架中,C#语言使用委托以及事件,可以很好的实现观察者模式。委托相当于“订阅清单”的角色,当目标中关联了该委托的事件被触发时,则委托将自动按序执行观察者注册于委托中的方法。
模型与观察者基类
我们把观察者模式的参与者都描述为派生自模型及观察者二个抽象基类的类。模型规划了事件,而观察者则规划了订阅及行为。
模型需要做的只是声明委托以及声明委托类型的事件。当然,还可以附加上封装了触发委托事件的方法。所有派生自模型的类都是具体目标,它们所要做的只是在适当的场合触发事件。(即发出“通知”)。
在观察者基类中,我们通过构造器将抽象的响应方法注册(订阅)于委托事件中。所有派生自观察者基类的类都是具体观察者。因为订阅行为已经在抽象基类完成,具体观察者需要做的只是通过覆盖观察者基类的方法去定义具体需要响应的行为,和通过构造器把需要观察的具体目标传递给基类构造器。
优点
通过对模型与观察者基类的分析可知,委托与事件的机制几乎消除了这两个模块之间的耦合,灵活性提高了很多。如果需要增加观察者,则只需要覆盖基类抽象方法及把观察目标传递给基类。
三、 事例
题目:猫大叫,两只老鼠开始逃跑,主人醒来,宝宝也醒来了并且哭了起来.
解决方案:
1. 建立模型(目标基类)
1
using System;
2
3
4
5
namespace DelegateEvent
6
7

{
8
9
/**//// <summary>
10
11
/// 在Observer Pattern(观察者模式)中,此类作为所有Subject(目标)的抽象基类
12
13
/// 所有要充当Subject的类(在此事例中为"猫")都继承于此类.
14
15
/// 我们说此类作为模型,用于规划目标(即发布方)所产生的事件,及提供触发
16
17
/// 事件的方法.
18
19
/// 此抽象类无抽象方法,主要是为了不能实例化该类对象,确保模式完整性.
20
21
/// 具体实施:
22
23
/// 1.声明委托
24
25
/// 2.声明委托类型事件
26
27
/// 3.提供触发事件的方法
28
29
/// </summary>
30
31
public abstract class ModelBase
32
33
{
34
35
public ModelBase()
36
37
{
38
39
}
40
41
/**//// <summary>
42
43
/// 声明一个委托,用于代理一系列"无返回"及"不带参"的自定义方法
44
45
/// </summary>
46
47
public delegate void SubEventHandler();
48
49
/**//// <summary>
50
51
/// 声明一个绑定于上行所定义的委托的事件
52
53
/// </summary>
54
55
public event SubEventHandler SubEvent;
56
57
58
59
/**//// <summary>
60
61
/// 封装了触发事件的方法
62
63
/// 主要为了规范化及安全性,除观察者基类外,其派生类不直接触发委托事件
64
65
/// </summary>
66
67
protected void Notify()
68
69
{
70
71
//提高执行效率及安全性
72
73
if(this.SubEvent!=null)
74
75
this.SubEvent();
76
77
78
79
}
80
81
}
82
83
}
84
85
2. 建立观察者基类(单行为,多行为)
//--------------------单行为---------------------
1
using System;
2
3
4
5
namespace DelegateEvent
6
7

{
8
9
/**//// <summary>
10
11
/// 在Observer Pattern(观察者模式)中,此类作为所有Observer(观察者)的抽象基类
12
13
/// 所有要充当观察者的类(在此事例中为"老鼠"和"人")都继承于此类.
14
15
/// 我们说此类作为观察者基类,用于规划所有观察者(即订阅方)订阅行为.
16
17
/// 在此事例中,规划了针对目标基类(ModelBase)中声明的"无参无返回"委托的一个
18
19
/// 方法(Response),并于构造该观察者时将其注册于具体目标(参数传递)的委托事件中.
20
21
/// 具体实施过程:
22
23
/// 1.指定观察者所观察的对象(即发布方).(通过构造器传递)
24
25
/// 2.规划观察者自身需要作出响应方法列表
26
27
/// 3.注册需要委托执行的方法.(通过构造器实现)
28
29
/// </summary>
30
31
public abstract class Observer
32
33
{
34
35
/**//// <summary>
36
37
/// 构造时通过传入模型对象,把观察者与模型关联,并完成订阅.
38
39
/// 在此确定需要观察的模型对象.
40
41
/// </summary>
42
43
/// <param name="childModel">需要观察的对象</param>
44
45
public Observer(ModelBase childModel)
46
47
{
48
49
//订阅
50
51
//把观察者行为(这里是Response)注册于委托事件
52
53
childModel.SubEvent+=new ModelBase.SubEventHandler(Response);
54
55
}
56
57
58
59
/**//// <summary>
60
61
/// 规划了观察者的一种行为(方法),所有派生于该观察者基类的具体观察者都
62
63
/// 通过覆盖该方法来实现作出响应的行为.
64
65
/// </summary>
66
67
public abstract void Response();
68
69
}
70
71
}
72
73
//-------------------多行为-------------------
1
using System;
2
3
namespace DelegateEvent
4
5

{
6
7
/**//// <summary>
8
9
/// 定义了另一个观察者基类.该观察者类型拥有两个响应行为.
10
11
/// 并在构造时将响应行为注册于委托事件.
12
13
/// (具体描述请参照另一观察者基类Observer)
14
15
/// </summary>
16
17
public abstract class Observer2
18
19
{
20
21
/**//// <summary>
22
23
/// 构造时通过传入模型对象,把观察者与模型关联,并完成订阅.
24
25
/// 在此确定需要观察的模型对象.
26
27
/// </summary>
28
29
/// <param name="childModel">需要观察的对象</param>
30
31
public Observer2(ModelBase childModel)
32
33
{
34
35
//订阅
36
37
//把观察者行为(这里是Response和Response2)注册于委托事件
38
39
childModel.SubEvent+=new ModelBase.SubEventHandler(Response);
40
41
childModel.SubEvent+=new ModelBase.SubEventHandler(Response2);
42
43
44
45
}
46
47
/**//// <summary>
48
49
/// 规划了观察者的二种行为(方法),所有派生于该观察者基类的具体观察者都
50
51
/// 通过覆盖该方法来实现作出响应的行为.
52
53
/// </summary>
54
55
public abstract void Response();
56
57
public abstract void Response2();
58
59
}
60
61
}
3.
建立具体目标
1
using System;
2
3
namespace DelegateEvent
4
5

{
6
7
/**//// <summary>
8
9
/// 此类为观察者模式中的具体目标(即具体发布方),其继承于模型.
10
11
/// 其中包含(调用)了在模型中被封装好的触发委托事件的方法.
12
13
/// </summary>
14
15
public class Cat:ModelBase
16
17
{
18
19
public Cat()
20
21
{
22
23
}
24
25