Net设计模式实例之观察者模式
一、观察者模式简介(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设计模式实例之观察者模式的更多相关文章
- Net设计模式实例系列文章总结
1 什么是设计模式 设计模式是对在软件设计过程中重复出现的问题提出了一种比较好的解决方案.正如一位专家所说:设计模式是对程序设计人员经常遇到的设计问题的可再现的解决方案(The Smalltalk C ...
- 设计模式 ( 十六 ) 观察者模式Observer(对象行为型)
设计模式 ( 十六 ) 观察者模式Observer(对象行为型) 1.概述 一些面向对象的编程方式,提供了一种构建对象间复杂网络互连的能力.当对象们连接在一起时,它们就可以相互提供服务和信息. 通常来 ...
- Java设计模式之《观察者模式》及应用场景
原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6513651.html 观察者模式,又可以称之为发布-订阅模式,观察者,顾名思义,就是一个监 ...
- Java设计模式百例 - 观察者模式
观察者(Observer)模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,主体对象的状态变化会通知所有观察者对象.观察者模式又叫做发布-订阅(Publish/Subscribe ...
- [head first 设计模式]第二章 观察者模式
[head first 设计模式]第二章 观察者模式 假如我们有一个开发需求--建造一个气象观测站展示系统.需求方给我们提供了一个WeatherObject对象,能够自动获得最新的测量数据.而我们要建 ...
- js设计模式之实现观察者模式实例代码
前端界面 html代码 <body> <select name="" id="select"> <option value=&qu ...
- Java设计模式10:观察者模式
观察者模式 观察者模式也叫作发布-订阅模式,也就是事件监听机制.观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态上发生变化时,会通知所有观察者对象,使他 ...
- 我理解设计模式C++实现观察者模式Observer Pattern
概述: 近期中国股市起起伏伏,当然了起伏就用商机,小明发现商机后果断想入市,买入了中国证券,他想在电脑client上,网页上,手机上,iPad上都能够查看到该证券的实时行情,这样的情况下我们应该怎么设 ...
- C#设计模式之十七观察者模式(Observer Pattern)【行为型】
一.引言 今天是2017年11月份的最后一天,也就是2017年11月30日,利用今天再写一个模式,争取下个月(也就是12月份)把所有的模式写完,2018年,新的一年写一些新的东西.今天我们开始讲& ...
随机推荐
- Linux命令之dig命令挖出DNS的秘密
=== [初次见面] 我相信使用nslookup的同学一定比使用dig的同学多,所以还是有必要花些时间给大家介绍一下dig的. dig,和nslookup作用有些类似,都是DNS查询工具. dig,其 ...
- mysql 导入数据到postgresql
创建PG的表脚本 DROP TABLE IF EXISTS "public"."t_resource_info"; CREATE TABLE "pub ...
- IEEEXtreme 10.0 - Food Truck
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme10.0 - Food Truck 题目来源 第10届IEEE极限编程大赛 https://www.hackerrank.c ...
- day4 使用yield实现单线程
一.yield生成器(yield) yield用来结束while循环,并且能够保持之前循环的状态,下一次调用的时候直接从yield开始执行,执行yield后面的程序,并且重新进行循环:另外,yield ...
- linux用户帐号管理/etcpasswd 和/etc/shadow文件
#学习鸟哥的linux私房菜 /etc/passwd的文件构造: dahu@dahu-OptiPlex-:~/myfile/VideoFile$ head /etc/passwd root:x:::r ...
- Educational Codeforces Round 51 (Rated for Div. 2) F - The Shortest Statement 倍增LCA + 最短路
F - The Shortest Statement emmm, 比赛的时候没有想到如何利用非树边. 其实感觉很简单.. 对于一个询问答案分为两部分求: 第一部分:只经过树边,用倍增就能求出来啦. 第 ...
- bzoj 1101 莫比乌斯反演
最裸的莫比乌斯 #include<bits/stdc++.h> #define LL long long #define fi first #define se second #defin ...
- 未能从程序集“Elmah”中加载类型“Elmah.ErrorLogModule”错误
项目名与Elmah重名了,以为是配置文件的问题,搞了好久.
- ES6 简介
1.全称: ECMA 标准,又称ES2015 JavaScript 是大家所了解的语言名称,但是这个语言名称是商标( Oracle 公司注册的商标).因此,JavaScript 的正式名称是 ECMA ...
- 七牛刷新接口PHP实现
<?php require_once '../autoload.php'; use Qiniu\Auth; use Qiniu\Http\Client; $accessKey = 'access ...