您的位置:首页 > 其它

观察者模式(Observer Pattern)

2008-08-04 15:12 267 查看
(1)应用:

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己

(2)角色:

* 抽象主题(Subject)角色:主题角色把所有对观察考对象的引用保存在一个聚集里,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象,主题角色又叫做抽象被观察者(Observable)角色,一般用一个抽象类或者一个接口实现。

* 抽象观察者(Observer)角色:为所有的具体观察者定义一个接口,在得到主题的通知时更新自己。这个接口叫做更新接口。抽象观察者角色一般用一个抽象类或者一个接口实现。在这个示意性的实现中,更新接口只包含一个方法(即Update()方法),这个方法叫做更新方法。

* 具体主题(ConcreteSubject)角色:将有关状态存入具体现察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色又叫做具体被观察者角色(Concrete Observable)。具体主题角色通常用一个具体子类实现。

* 具体观察者(ConcreteObserver)角色:存储与主题的状态自恰的状态。具体现察者角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态相协调。如果需要,具体现察者角色可以保存一个指向具体主题对象的引用。具体观察者角色通常用一个具体子类实现。

(.NET中提供了Delegate与Event机制,我们可以利用这种机制简化Observer模式。)

(3)故事:

此时,玉皇大帝和一群神仙已来到南天门观战,只见二郎神率领梅山六兄弟围着孙悟空好一阵穷追猛打。

太上老君说:“让我来助二郎神一功。”说着捋起衣袖,从左胳膊上取下一只“金钢套”,

照准猴王的脑袋扔了下去。猴王苦战之中,来不及躲闪,被“金钢套”打中了天灵,跌了一跤。

正待爬将起来就跑,被二郎神豢养的哮天犬赶上来,一口咬住了腿肚子。

二郎神和梅山六兄弟一拥而上,终于捉住了这个不可一世的妖猴。

(4)分析:

观察者模式定义了一种一(猴子)对多(观战团成员)的依赖关系,让多个观察者对象(观战团成员)同时监听某一个主题对象(猴子)。这个主题对象(猴子)在状态上发生变化时,会通知所有观察者对象(观战团成员工),使它们能够自动更新自己

(5)实现

using System;

using System.Collections.Generic;

public delegate void 委托通知观察者(状态持有者 观察对象);

public class 状态持有者

{

/// <summary>

/// 私有状态

/// </summary>

private string _状态;

/// <summary>

/// 只读

/// </summary>

public string 状态 { get { return _状态; } }

public 状态持有者()

{

_状态 = "正在战斗";

}

/// <summary>

/// 战斗过程

/// </summary>

public void 战斗过程开始()

{

//通知观战团猴子的被始状态

设置状态(_状态);

//变更状态,也会引发通知

Console.WriteLine("二郎神率领梅山六兄弟围着孙悟空好一阵穷追猛打");

设置状态("被穷追猛打");

}

//为了示例,让你们直接修改我的状态好了

//其实你扔金钢套,不一定能打中我

//另外我也不怕狗

public void 设置状态(string 状态)

{

_状态 = 状态;

if (通知观察者 != null)

{

Console.WriteLine("现在还有{0}个观察者在偷偷看我" , 通知观察者.GetInvocationList().Length);

通知观察者(this);

}

}

public event 委托通知观察者 通知观察者;

}

/// <summary>

/// 抽象观察者,除了玉帝,其它人不允许你干看着

///对了玉帝也不行,你不出力也要喊两嗓子,以示你是一个观察者

/// </summary>

public inte***ce 观察者

{

void 行动(状态持有者 观察对象);

}

/// <summary>

/// 对不起,我只会说不会做

/// </summary>

public class 玉帝 : 观察者

{

public void 行动(状态持有者 观察对象)

{

Console.WriteLine("玉帝:我看到猴子" + 观察对象.状态);

}

}

public class 太上老君 : 观察者

{

public void 行动(状态持有者 观察对象)

{

if (观察对象.状态 == "正在战斗")

{

Console.WriteLine("太上老君:我在等待时机");

}

//该出手时就出手呀

else if (观察对象.状态 == "被穷追猛打")

{

Console.WriteLine("太上老君:我扔下金钢套,打中了猴子");

观察对象.通知观察者 -= new 委托通知观察者(行动);

Console.WriteLine("太上老君:任务完成,退出观察者行列");

观察对象.设置状态("跌了一跤");

}

}

}

/// <summary>

/// 其中的一名观察者,为了等猴子跌一跤

/// </summary>

public class 哮天犬 : 观察者

{

public void 行动(状态持有者 观察对象)

{

if (观察对象.状态 == "跌了一跤")

{

Console.WriteLine("哮天犬赶上来,一口咬住了腿肚子");

观察对象.通知观察者 -= new 委托通知观察者(行动);

观察对象.设置状态("咬住了腿肚子");

}

}

}

/// <summary>

///作用与哮天犬同

///为了讲完这个故事

/// </summary>

public class 二郎神和梅山六兄弟 : 观察者

{

public void 行动(状态持有者 观察对象)

{

if (观察对象.状态 == "咬住了腿肚子")

{

Console.WriteLine("二郎神和梅山六兄弟一拥而上,终于捉住了这个不可一世的妖猴");

观察对象.通知观察者 -= new 委托通知观察者(行动);

观察对象.设置状态("被捉住了");

}

}

}

public class MyClass

{

public static void Main()

{

//天庭观战团有四名成员

观察者[] 天庭观战团 = new 观察者[]{

new 玉帝(),

new 太上老君(),

new 哮天犬(),

new 二郎神和梅山六兄弟()

};

//实例化状态持有者,这儿就是孙悟空

状态持有者 孙悟空 = new 状态持有者();

//孙悟空要知会所有对自己状态感兴趣的成员

//有点笨,你不通知它们不就好了

foreach (观察者 实例化的观察者 in 天庭观战团)

{

孙悟空.通知观察者 += new 委托通知观察者(实例化的观察者.行动);

}

//开始打斗

孙悟空.战斗过程开始();

Console.Read();

}

}

(6)结果

现在还有4个观察者在偷偷看我

玉帝:我看到猴子正在战斗

太上老君:我在等待时机

二郎神率领梅山六兄弟围着孙悟空好一阵穷追猛打

现在还有4个观察者在偷偷看我

玉帝:我看到猴子被穷追猛打

太上老君:我扔下金钢套,打中了猴子

太上老君:任务完成,退出观察者行列

现在还有3个观察者在偷偷看我

玉帝:我看到猴子跌了一跤

哮天犬赶上来,一口咬住了腿肚子

现在还有2个观察者在偷偷看我

玉帝:我看到猴子咬住了腿肚子

二郎神和梅山六兄弟一拥而上,终于捉住了这个不可一世的妖猴

现在还有1个观察者在偷偷看我

玉帝:我看到猴子被捉住了
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: