一、观察者模式简介(Brief Introduction)

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

二、解决的问题(What To Solve)

当一个对象的改变需要同时改变其他对象的时候,而且不知道有多少对象有待改变时,应该考虑使用观察者模式。

观察者模式所做的工作其实就是解除耦合,让耦合的双方都依赖于抽象,而不是依赖于具体,从而使的各自的变化都不会影响另一边的变化。

三、观察者模式分析(Analysis)

1、观察者模式结构

Subject:它把所有对观察者对象的引用保存在一个聚集里面,每个主题都可以有任意数量的观察者,抽象主题提供一个接口,可以增加和删除观察者对象。

public void Notify()

{

foreach(Observer o in observers)

{

o.Update();

}

ConcreteSubject: 具体的主题,将有关状态存入具体观察者对象,在具体主题的内部状态改变时,给所有登记国的观察者发出通知。

Observer:抽象观察者,为所有的具体观察者定义一个接口,在得到主题的通知时更新自己

ConcreteObserver:具体观察者,实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态相协调

2、源代码

1、Subject类,主题或者抽象通知者

public abstract class Subject

{

private IList<Observer> observers = new List<Observer>();

/// <summary>

/// 添加观察者

/// </summary>

/// <param name="observer">观察者</param>

public void Attach(Observer observer)

{

observers.Add(observer);

}

/// <summary>

/// 移除观察者

/// </summary>

/// <param name="observer">观察者</param>

public void Detach(Observer observer)

{

observers.Remove(observer);

}

/// <summary>

/// 通知观察者

/// </summary>

public void Notify()

{

foreach (Observer o in observers)

{

o.Update();

}

}

}

2、ConcreteSubject类,具体主题或者具体通知者

public class ConcreteSubject:Subject

{

private string _subjectState;

/// <summary>

/// 具体被观察者状态

/// </summary>

public string SubjectState

{

get { return _subjectState; }

set { _subjectState = value; }

}

}

3、Observer抽象观察者,为所有的具体观察者定义一个接口

public abstract class Observer

{

public abstract void Update();

}

4、ConcreteObserver具体观察者

/// <summary>

/// 具体观察者,实现抽象观察者角色所要求的更新接口

/// 以便使本身的状态与主题的状态相协调

/// </summary>

public class ConcreteObserver:Observer

{

private string name;

private string observerState;

private ConcreteSubject subject;

public ConcreteSubject Subject

{

get { return subject; }

set { subject = value; }

}

public ConcreteObserver(ConcreteSubject subject,string name)

{

this.subject = subject;

this.name = name;

}

public override void Update()

{

observerState = subject.SubjectState;

Console.WriteLine("观察者{0}的新状态是{1}",name,observerState);

}

}

5、客户端代码

static void Main(string[] args)

{

ConcreteSubject cs = new ConcreteSubject();

cs.Attach(new ConcreteObserver(cs,"James"));

cs.Attach(new ConcreteObserver(cs,"Jane"));

cs.SubjectState="OK";

cs.Notify();

Console.Read();

}

3、程序运行结果

四.观察者实例分析(Example)

1、场景

假设有一股票开盘价格16.50元,自从上市以来价格是不断下降,而且以1.00元的速度下降。

在股票降到12.00元时,股民灵动生活买入了股票。

在股票降到8.05元时,股民Jane买了股票。

2、观察者实例结构

Stock,抽象通知者

定义了委托PriceChangedHandler ,调用了事件参数StockDetailsArgs 。

声明了事件PriceChanged.

股票在下跌的过程中调用方法OnPriceChanged ,通过此方法触发事件PriceChanged 。

AttachEvent 方法用来添加观察者到对象

StockDetailArgs,事件参数继承于EventArgs类,有树形CurrentPrice用来专递价格数据

接口IObserver和具体观察者Observer类:

Stoc_PriceChanged方法:当股票在以1.00元降价的过程中调用此方法。当价格降到符合购买者价格,而且股票没有被其他人购买的情况时,执行购买行为。

开盘价格:16.50

收盘价格:5.50

当价格降到12.00时,观察者灵动生活买入此股票

当价格降到8.05时,观察者Jane买入此股票

3、代码

1、Stock股票类

public class Stock

{

private double _openPrice;

private double _closePrice;

public delegate void PriceChangedHandler(object sender, StockDetailArgs e);

public event PriceChangedHandler PriceChanged;

public double OpenPrice

{

get { return _openPrice; }

set { _openPrice = value; }

}

public double ClosePrice

{

get { return _closePrice; }

set { _closePrice = value; }

}

public void StartTrading()

{

double current;

//Current price decrements by $1.00 as the stock is traded

current = OpenPrice;

while (current > ClosePrice)

{

//Stock is falling in increments of $1.00

current = current - 1.00;

//Call the method to raise the event

OnPriceChanged(current);

//Simulate a delay of 2000ms between market price updates

System.Threading.Thread.Sleep(2000);

}

}

protected void OnPriceChanged(double currentMarketPrice)

{

//Any handlers attached to this event?

if (PriceChanged != null)

{

StockDetailArgs args = new StockDetailArgs();

args.CurrentPrice = currentMarketPrice;

Console.WriteLine("当前股票价格是:" + args.CurrentPrice.ToString());

////Raise the event

PriceChanged(this, args);

}

}

/// <summary>

/// 添加观察者

/// </summary>

/// <param name="observer">观察者</param>

public void AttachEvent(IObserver observer)

{

PriceChanged += new PriceChangedHandler(observer.Stoc_PriceChanged);

}

}

 

2、事件参数StockDetailArgs

public class StockDetailArgs: EventArgs

{

private double _currentPrice;

public double CurrentPrice

{

get { return _currentPrice; }

set { _currentPrice = value; }

}

}

 

3、观察者接口IObserver

public interface IObserver

{

void Stoc_PriceChanged(object sender, StockDetailArgs e);

}

 

4、具体观察者Observer

public class Observer : IObserver

{

private string _investorName;

private double _buyPrice;

private Stock _stoc;

private bool _hasBoughtStock = false;

public string InvestorName

{

get { return _investorName; }

set { _investorName = value; }

}

public double BuyPrice

{

get { return _buyPrice; }

set { _buyPrice = value; }

}

public Stock Stoc

{

get { return _stoc; }

set { _stoc = value; }

}

public Observer(string investorName, double buyPrice)

{

this.InvestorName = investorName;

this.BuyPrice = buyPrice;

}

public void Stoc_PriceChanged(object sender, StockDetailArgs e)

{

if (e.CurrentPrice <= BuyPrice && _hasBoughtStock == false)

{

Console.WriteLine(string.Format("{0}在价格Price ={1}时买进了股票。",InvestorName,e.CurrentPrice));

_hasBoughtStock = true;

}

}

}

 

5、客户端代码

static void Main(string[] args)

{

Stock stock = new Stock();

stock.OpenPrice = 16.50;

stock.ClosePrice = 5.50;

Observer james = new Observer("灵动生活", 12.00);

Observer jane = new Observer("jane",8.05);

stock.AttachEvent(james);

stock.AttachEvent(jane);

stock.StartTrading();

Console.Read();

}

 

4、程序运行结果

五、总结(Summary)

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态发生变化的时,会通知所有观察者对象,使他们能够自动更新自己。解决的是“当一个对象的改变需要同时改变其他对象的时候”问题。

Net设计模式实例之观察者模式的更多相关文章

  1. Net设计模式实例系列文章总结

    1 什么是设计模式 设计模式是对在软件设计过程中重复出现的问题提出了一种比较好的解决方案.正如一位专家所说:设计模式是对程序设计人员经常遇到的设计问题的可再现的解决方案(The Smalltalk C ...

  2. 设计模式 ( 十六 ) 观察者模式Observer(对象行为型)

    设计模式 ( 十六 ) 观察者模式Observer(对象行为型) 1.概述 一些面向对象的编程方式,提供了一种构建对象间复杂网络互连的能力.当对象们连接在一起时,它们就可以相互提供服务和信息. 通常来 ...

  3. Java设计模式之《观察者模式》及应用场景

    原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6513651.html 观察者模式,又可以称之为发布-订阅模式,观察者,顾名思义,就是一个监 ...

  4. Java设计模式百例 - 观察者模式

    观察者(Observer)模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,主体对象的状态变化会通知所有观察者对象.观察者模式又叫做发布-订阅(Publish/Subscribe ...

  5. [head first 设计模式]第二章 观察者模式

    [head first 设计模式]第二章 观察者模式 假如我们有一个开发需求--建造一个气象观测站展示系统.需求方给我们提供了一个WeatherObject对象,能够自动获得最新的测量数据.而我们要建 ...

  6. js设计模式之实现观察者模式实例代码

    前端界面 html代码 <body> <select name="" id="select"> <option value=&qu ...

  7. Java设计模式10:观察者模式

    观察者模式 观察者模式也叫作发布-订阅模式,也就是事件监听机制.观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态上发生变化时,会通知所有观察者对象,使他 ...

  8. 我理解设计模式C++实现观察者模式Observer Pattern

    概述: 近期中国股市起起伏伏,当然了起伏就用商机,小明发现商机后果断想入市,买入了中国证券,他想在电脑client上,网页上,手机上,iPad上都能够查看到该证券的实时行情,这样的情况下我们应该怎么设 ...

  9. C#设计模式之十七观察者模式(Observer Pattern)【行为型】

    一.引言   今天是2017年11月份的最后一天,也就是2017年11月30日,利用今天再写一个模式,争取下个月(也就是12月份)把所有的模式写完,2018年,新的一年写一些新的东西.今天我们开始讲& ...

随机推荐

  1. centos6下mysql的主从复制的配置

    2015年9月17日 23:00:36 update 想要好好了解mysql复制,还是去看看<高性能MySQL>(第三版)好了,上面说的比较详细. =========== 在本地用virt ...

  2. 第 17 章 使用API

    在本章中,我们将学习如何编写一个独立的程序,并对其获取的数据进行可视化.这个程序将使用Web应用编程接口(API)自动请求网站的特定信息而不是整个网页,再对这些信息进行可视化.由于这样编写的程序始终使 ...

  3. 常用的scrapy setting

    原文请参考    Scrapy 爬虫入门教程十三 Settings(设置), 讲的很详细 官网参考  Settings 设置 Scrapy 设置允许您自定义所有 Scrapy 组件的行为,包括核心,扩 ...

  4. 浅谈css中浮动和清除浮动带来的影响

    有很多时候,我们都会用到浮动,而我们有时候对浮动只是一知半解,却不是太清楚它到底是怎么回事,不知道各位有没有和我一样的感觉,只知道用它,却不知道它到底是怎么回事,所以,在学习的过程中,就要把一个概念不 ...

  5. bzoj 1880 最短路径图

    #include<bits/stdc++.h> #define LL long long #define fi first #define se second #define mk mak ...

  6. windows下重置mysql的root密码方法介绍(转)

    自己在内网操作的,遇到了一些的问题,其中一个是需要重置密码的,所以网上找了两篇文章,都有一些借鉴的地方. 版本mysql5.7.2,linux系统 除了参考文章还有几点说明: service mysq ...

  7. Unity 游戏开发技巧集锦之使用忍者飞镖创建粒子效果

    Unity 游戏开发技巧集锦之使用忍者飞镖创建粒子效果 使用忍者飞镖创建粒子效果 游戏中,诸如烟.火.水滴.落叶等粒子效果,都可以使用粒子系统(particle system)来实现.例如,<明 ...

  8. SpringBoot学习(六)

    1.pom 文件 <?xml version="1.0" encoding="utf-8"?> <dependencies> <d ...

  9. 【UOJ #103】【APIO 2014】Palindromes

    http://uoj.ac/problem/103 由manacher得:本质不同的回文串只有\(O(n)\)个. 用manacher求出所有本质不同的回文串,对每个本质不同的回文串,在后缀自动机的p ...

  10. [BZOJ5028]小Z的加油店

    [BZOJ5028]小Z的加油店 题目大意: 一个长度为\(n(n\le10^5)\)的数列,\(m(m\le10^5)\)次操作,支持区间加和区间\(\gcd\). 思路: 线段树维护差分,\(\g ...