简介:

观察者模式,也称为订阅-发布模式,定义对象间的一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖他的对象都得到通知并被自动更新。

主要由以下几个部分组成:

a.Subject目标对象。它具有以下特征:

一个目标可以被多个观察者订阅

提供订阅和取消订阅的方法

当目标对象状态发生变化时,通知所有订阅者。

把Subject独立出来是因为他提供了观察者模式中常见的三个特征,每个观察者模式都是这样,可以抽象出来。具体的单独逻辑可放在ConcreteSubject具体的目标实现对象中。

b.Observer定义观察者的接口。提供方法(一般为接口),当目标对象发生变化通知过来,做对应的响应操作。可以在该方法中调用Subject目标对象,以获取目标对象的数据。

c.ConcreteSubject具体的目标实现对象。可以维护具体对象的一些个性化属性,如目标状态等。

d.ConcreteObserver观察者的具体实现对象。提供处理目标对象变化的具体响应操作。

具体实例:

报社发行报纸,广大读者可以订报,也可以取消订报。报社维护订阅者的名单,一旦有新的报纸印刷出来,及时发布给广大读者。

Subject目标对象:提供观察者模式的基本功能

上面的例子中,广大读者都在观察同一个报社对象,这个报社对象就是被观察的目标。一般建议在目标接口名称后面加上Subject

/// <summary>
/// Subject目标对象
/// </summary>
public class NewsPaperSubject
{
/// <summary>
/// 维护有效订阅者的列表。
/// </summary>
protected List<ReaderObserver> observerList = new List<ReaderObserver>(); /// <summary>
/// 提供公共接口,供注册订阅者
/// </summary>
/// <param name="aObserver"></param>
public void Attach(ReaderObserver aObserver)
{
observerList.Add(aObserver);
} /// <summary>
/// 提供公共接口,供取消注册订阅者
/// </summary>
/// <param name="aObserver"></param>
public void Detach(ReaderObserver aObserver)
{
observerList.Remove(aObserver);
} /// <summary>
/// 目标对象发生变化时,通知所有订阅的观察者
/// </summary>
protected void NotifyAllObservers()
{
foreach (ReaderObserver observer in observerList)
{
observer.update(this);
}
}
}

上面的例子中,仅实现了报社的基本功能,并没有定义报社出版报纸等功能,是为了这个更通用。具体的逻辑放在ConcreteSubject中。

ConcreteSubject具体的目标实现对象:真正的模板对象,日报

日报,在原来基类上新增属性:报纸的出版日期。修改了出版日期,相当于重新出版了新报纸,此时需要通知所有读者。

 
/// <summary>
/// ConcreteSubject具体的目标实现对象 -日报
/// </summary>
public class DaliyNewsPaper : NewsPaperSubject
{
public string Date
{
get { return _date; }
set
{
_date = value;
//改了出版日期,相当于重新出版了新报纸,此时需要通知所有读者。
NotifyAllObservers();
}
}
private string _date;
}
Observer定义观察者的接口:观察者的接口或者抽象类。
一般建议在观察者接口后面跟Observer
接口更新方法,建议命名以Update。
观察者订阅好报纸以后,在家里等着收报纸即可,没有其他的功能需要实现。这里先抽象出观察者接口,然后再后续类中具体实现。
/// <summary>
/// Observer定义观察者的接口
/// </summary>
public abstract class ReaderObserver
{
/// <summary>
/// 目标对象发生变化通知过来,响应操作接口方法。
/// <param name="aSubject">推送过来的目标对象</param>
/// </summary>
public abstract void update(NewsPaperSubject aSubject);
}
ConcreteSubject具体的目标实现对象:真正的观察者,读者
/// <summary>
/// ConcreteObserver观察者的具体实现对象 -读者
/// </summary>
public class DailyNewsPaperReader : ReaderObserver
{
public string Name
{
get { return _name; }
set { _name = value; }
}
private string _name; public override void update(NewsPaperSubject aSubject)
{
Console.WriteLine("{0}已经收到{1}的报纸", _name, (aSubject as DaliyNewsPaper).Date);
}
}
测试代码如下:
public class ObserverTest
{
public static void Main(string[] args)
{
//创建目标对象:报纸
DaliyNewsPaper newsPaper = new DaliyNewsPaper(); //创建观察者对象:张三
DailyNewsPaperReader observer1 = new DailyNewsPaperReader();
observer1.Name = "张三";
//张三订报纸
newsPaper.Attach(observer1); //创建观察者对象:李四
DailyNewsPaperReader observer2 = new DailyNewsPaperReader();
observer2.Name = "李四";
//李四订报纸
newsPaper.Attach(observer2); //2013-10-24报纸出版,张三、李四收到报纸
Console.WriteLine("2013-10-24报纸出版了!!!");
newsPaper.Date = "2013-10-24";
Console.WriteLine(); //李四取消订报
newsPaper.Detach(observer2); //2013-10-25报纸出版,张三收到报纸
Console.WriteLine("2013-10-25报纸出版了!!!");
newsPaper.Date = "2013-10-25"; Console.ReadLine();
}
}
补充分析:
1.观察者模式一般都是目标对象跟观察者一对多的关系,即一个目标有多个观察者。
对于观察者需要观察多个目标的情况下,可以修改观察者的UPDATE方法,根据传入参数决定是哪个目标的变化,或者干脆定义不同名字的UPDATE方法来对应不同的目标调用。
2.观察者模式的变例:区别对待观察者
比如说:某公司的请假制度为,请假半天以下的,通知下项目经理和项目组内成员即可,请假3天以下(含3天)的,还需要通知部门经理,请假3天以上的,还需要通知总经理。
实例中,请假对象是一个目标对象,项目经理、项目组成员、部门经理、总经理都是观察者。但是不同的是,并不是每次请假都需要通知上述所有人,需要根据请假的天数来决定通知的范围。
如果使用观察者模式?怎么处理?
参考代码如下:
protected void NotifyObservers()
{
foreach (ReaderObserver observer in observerList)
{
//不管请多少天,都需要通知项目经理和项目组成员
observer.getJob().equals("项目经理").Update(this);
observer.getJob().equals("项目组成员").Update(this); if (this.Day > 0.5)
observer.getJob().equals("部门经理").Update(this); if(this.Death >3)
observer.getJob().equals("总经理").Update(this);
}
}
此时具体目标对象需要重写抽象目标对象的通知方法,在方法内部根据目标的属性来决定被通知的观察者。
 
 

设计模式-观察者模式(Observer)的更多相关文章

  1. java设计模式--观察者模式(Observer)

    java设计模式--观察者模式(Observer) java设计模式--观察者模式(Observer) 观察者模式的定义: 定义对象间的一种一对多的依赖关系.当一个对象的状态发生改变时,所有依赖于它的 ...

  2. 设计模式-观察者模式(Observer Pattern)

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

  3. 设计模式 - 观察者模式(Observer Pattern) 详细说明

    观察者模式(Observer Pattern) 详细说明 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26583157 版权全部 ...

  4. 设计模式 - 观察者模式(Observer Pattern) 详细解释

    观察者模式(Observer Pattern) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26583157 版权全部 ...

  5. 设计模式 - 观察者模式(Observer Pattern) Java内置 用法

    观察者模式(Observer Pattern) Java内置 用法 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26601659 ...

  6. 设计模式--观察者模式Observer(对象行为型)

    一.观察者模式 观察者模式是在对象之间定义一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象都会收到通知,并自动更新.观察者模式也被称之为:主题-观察者模式,发布-订阅模式,前者是一,后者是多. ...

  7. 大话设计模式--观察者模式 Observer -- C++ 实现实例

    大话设计模式--1.观察者模式: 定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态发生变化时,会通知所有的 观察者对象,使他们能够自动更新自己. 使用场合: 当一 ...

  8. [工作中的设计模式]观察者模式observer

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

  9. C#设计模式——观察者模式(Observer Pattern)1

    一.概述在软件设计工作中会存在对象之间的依赖关系,当某一对象发生变化时,所有依赖它的对象都需要得到通知.如果设计的不好,很容易造成对象之间的耦合度太高,难以应对变化.使用观察者模式可以降低对象之间的依 ...

  10. C#设计模式——观察者模式(Observer Pattern)

    一.概述在软件设计工作中会存在对象之间的依赖关系,当某一对象发生变化时,所有依赖它的对象都需要得到通知.如果设计的不好,很容易造成对象之间的耦合度太高,难以应对变化.使用观察者模式可以降低对象之间的依 ...

随机推荐

  1. get/close not same thread Druid 连接池一个设置

    我就郁闷了,1000W+数据审核每次总是到一半就出这么个错,仔细找找原来是一个配置项的小问题,removeAbandonedTimeout 这个代表你从连接池取出一个连接多少秒之后你还没还回来,那就强 ...

  2. Yii 发送电子邮件

    yii 收发邮件 ------------------------------------------------------------------------------------------- ...

  3. jquery validate.addMethod 正则表达式

    $(document).ready(function () { /* 设置默认属性 */ $.validator.setDefaults( { submitHandler: function (for ...

  4. oracle decode

    decode()函数简介: 主要作用:将查询结果翻译成其他值(即以其他形式表现出来,以下举例说明): 使用方法: Select decode(columnname,值1,翻译值1,值2,翻译值2,.. ...

  5. 91、sendToTarget与sendMessage

    Message msg = handler.obtainMessage();               msg.arg1 = i;               msg.sendToTarget(); ...

  6. 四个排名函数(row_number、rank、dense_rank和ntile)的比较

    排名函数是SQL Server2005新加的功能.在SQL Server2005中有如下四个排名函数: 1.row_number 2.rank 3.dense_rank 4.ntile 下面分别介绍一 ...

  7. python中报错总结

    python标准异常 异常名称 描述     BaseException 所有异常的基类 SystemExit 解释器请求退出 KeyboardInterrupt 用户中断执行(通常是输入^C) Ex ...

  8. .net中三种数据类型转换区别((int),Int32.Parse() 和 Convert.toInt32() )

    (typename)valuename,是通用方法: Convert类提供了灵活的类型转换封装: Parse方法,适用于向数字类型的转换. 例如,(int),Int32.Parse() 和 Conve ...

  9. iOS UIButton 设置图片文字垂直排列

    后面经过测试,如果button的文字长度变更,会导致图片位置变化,经过多次修改UIEdgeInsets的值也没有达到期望效果,最终采用集成UIButton类,重写layoutSubviews函数实现, ...

  10. [转]整理索引碎片,提升SQL Server速度

    数据库表A有十万条记录,查询速度本来还可以,但导入一千条数据后,问题出现了.当选择的数据在原十万条记录之间时,速度还是挺快的:但当选择的数据在这一千条数据之间时,速度变得奇慢. 凭经验,这是索引碎片问 ...