SAL

  博客园  :: 首页  :: 新随笔  :: 订阅 订阅  :: 管理

观察者模式(Observer Pattern)

Posted on 2008-08-04 15:12  SAL  阅读(333)  评论(0编辑  收藏  举报
(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个观察者在偷偷看我
玉帝:我看到猴子被捉住了