C#设计模式--观察者模式(发布-订阅模式)
设计模式:
观察者模式(Observer Pattern)
简单介绍:
观察者模式(Observer Pattern)是设计模式中行为模式的一种,它解决了上述具有一对多依赖关系的对象的重用问题。此模式的参与者分为两大类,一类是被观察的目标,另一类是观察该目标的观察者们。正因为该模式是基于“一对多”的关系,所以该模式一般是应用于由一个目标对象和N个观察者对象组成(当然也可以扩展为有多个目标对象,但我们现在只讨论前者)的场合。当目标对象的状态发生改变或做出某种行为时,正在观察该目标对象的观察者们将自动地、连锁地作出相应的响应行为。
模式中具有的角色
- 抽象主题(Subject):它把所有观察者对象的引用保存到一个聚集里,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象。
- 具体主题(ConcreteSubject):将有关状态存入具体观察者对象;在具体主题内部状态改变时,给所有登记过的观察者发出通知。
- 抽象观察者(Observer):为所有的具体观察者定义一个接口,在得到主题通知时更新自己。
- 具体观察者(ConcreteObserver):实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题状态协调。
举例子:博客的例子就是一个观察者模式,比如你关注一些作者的博客,当作者有博客发布时候,你就会收到一条该作者发布的博客的消息。
抽象主题:Blog 博客
具体主题:MyBlog 的博客
抽象观察者:IObserver
具体的观察者:Observer
观察者模式类图:

观察者模式C#代码举例
订阅博客抽象类
/*****************************************************
* ProjectName: _11DesignPattern_Observer
* Description:
* ClassName: Blog
* CLRVersion: 4.0.30319.18408
* Author: JiYF
* NameSpace: _11DesignPattern_Observer
* MachineName: JIYONGFEI
* CreateTime: 2017/5/23 17:08:10
* UpdatedTime: 2017/5/23 17:08:10
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_Observer
{
/// <summary>
/// 订阅博客抽象类
/// </summary>
public abstract class Blog
{
/// <summary>
/// 保存订阅者列表
/// </summary>
private List<IObserver> observers = new List<IObserver>(); /// <summary>
/// 博主名
/// </summary>
public string BlogName { get; set; } /// <summary>
/// 博客标题
/// </summary>
public string BlogTitle { get; set; } /// <summary>
/// 博客信息
/// </summary>
public string BlogInfo { get; set; } /// <summary>
/// 博客构造函数
/// </summary>
/// <param name="blogTitle">博客标题</param>
/// <param name="blogInfo">博客信息</param>
public Blog(string name,string blogTitle,string blogInfo)
{
this.BlogName = name;
this.BlogTitle = blogTitle;
this.BlogInfo = blogInfo;
} /// <summary>
/// 添加一个订阅者
/// </summary>
/// <param name="observer">具体的订阅者对象</param>
public void AddObserver(IObserver observer)
{
if (observers.Contains(observer))
{
return;
}
observers.Add(observer);
} /// <summary>
/// 删除一个订阅者
/// </summary>
/// <param name="observer">具体的订阅者对象</param>
public void RemoveObserver(IObserver observer)
{
if (observers.Contains(observer))
{
observers.Remove(observer);
}
} /// <summary>
/// 发布博客通知
/// </summary>
public void PublishBlog()
{
//遍历通知每一个订阅者
foreach (IObserver ob in observers)
{
if (ob != null)
{
ob.Receive(this);
}
}
} }
}
具体的博客类
/*****************************************************
* ProjectName: _11DesignPattern_Observer
* Description:
* ClassName: JiYFBlog
* CLRVersion: 4.0.30319.18408
* Author: JiYF
* NameSpace: _11DesignPattern_Observer
* MachineName: JIYONGFEI
* CreateTime: 2017/5/23 17:21:23
* UpdatedTime: 2017/5/23 17:21:23
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_Observer
{ /// <summary>
/// 具体的订阅博客类
/// </summary>
public class JiYFBlog :Blog
{
public JiYFBlog(string name,string blogTitile, string blogInfo)
: base(name,blogTitile,blogInfo)
{ }
}
}
订阅者接口类
/*****************************************************
* ProjectName: _11DesignPattern_Observer
* Description:
* ClassName: IObserver
* CLRVersion: 4.0.30319.18408
* Author: JiYF
* NameSpace: _11DesignPattern_Observer
* MachineName: JIYONGFEI
* CreateTime: 2017/5/23 17:09:28
* UpdatedTime: 2017/5/23 17:09:28
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_Observer
{
/// <summary>
/// 订阅者接口
/// </summary>
public interface IObserver
{ void Receive(Blog blog);
}
}
具体的订阅者类:
/*****************************************************
* ProjectName: _11DesignPattern_Observer
* Description:
* ClassName: Observer
* CLRVersion: 4.0.30319.18408
* Author: JiYF
* NameSpace: _11DesignPattern_Observer
* MachineName: JIYONGFEI
* CreateTime: 2017/5/23 17:23:36
* UpdatedTime: 2017/5/23 17:23:36
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_Observer
{
/// <summary>
/// 具体的订阅者类
/// </summary>
public class Observer :IObserver
{
/// <summary>
/// 订阅者名字
/// </summary>
private string m_Name;
public string Name
{
get { return m_Name; }
set { m_Name = value; }
} /// <summary>
/// 订阅者构造函数
/// </summary>
/// <param name="name">订阅者名字</param>
public Observer(string name)
{
this.m_Name = name;
} /// <summary>
/// 订阅者接受函数
/// </summary>
/// <param name="blog"></param>
public void Receive(Blog blog)
{
Console.WriteLine("订阅者:\"{0}\"观察到了:{1}发布的一篇博客,标题为:{2},内容为:{3}",Name,blog.BlogName,blog.BlogTitle,blog.BlogInfo);
}
}
}
测试代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_Observer
{
class Client
{
static void Main(string[] args)
{
Console.WriteLine("--全部订阅者--");
//创建一个JiYF的博客
Blog jiyfBlog = new JiYFBlog("JiYF笨小孩","丑小鸭","丑小鸭的故事"); //创建订阅者
Observer obsZhangsan = new Observer("张三");
Observer obsLiSi = new Observer("李四");
Observer obsWangwu = new Observer("王五"); //添加对JiYF博客的订阅者
jiyfBlog.AddObserver(obsZhangsan);
jiyfBlog.AddObserver(obsLiSi);
jiyfBlog.AddObserver(obsWangwu); //通知订阅者
jiyfBlog.PublishBlog(); Console.WriteLine("--移除订阅者张三--");
jiyfBlog.RemoveObserver(obsZhangsan);
jiyfBlog.PublishBlog();
Console.ReadLine();
}
}
}
运行结果:

通常在C#代码中,观察者模式一般采用委托来实现如下所示
委托实现观察者模式
抽象博客类:
/*****************************************************
* ProjectName: _11DesignPattern_ObserverDelegateTest
* Description:
* ClassName: Blog
* CLRVersion: 4.0.30319.42000
* Author: JiYF
* NameSpace: _11DesignPattern_ObserverDelegateTest
* MachineName: JIYF_PC
* CreateTime: 2017/5/23 20:48:00
* UpdatedTime: 2017/5/23 20:48:00
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_ObserverDelegateTest
{
/// <summary>
/// 委托充当订阅者接口类
/// </summary>
/// <param name="sender"></param>
public delegate void NotifyEventHandler(object sender); /// <summary>
/// 订阅博客基类
/// </summary>
public class Blog
{
public NotifyEventHandler NotifyEvent;
/// <summary>
/// 博主名
/// </summary>
public string BlogName { get; set; } /// <summary>
/// 博客标题
/// </summary>
public string BlogTitle { get; set; } /// <summary>
/// 博客信息
/// </summary>
public string BlogInfo { get; set; } /// <summary>
/// 博客构造函数
/// </summary>
/// <param name="blogTitle">博客标题</param>
/// <param name="blogInfo">博客信息</param>
public Blog(string name, string blogTitle, string blogInfo)
{
this.BlogName = name;
this.BlogTitle = blogTitle;
this.BlogInfo = blogInfo;
} /// <summary>
/// 绑定订阅事件
/// </summary>
/// <param name="ob">订阅者</param>
public void AddObserver(NotifyEventHandler observer)
{
NotifyEvent += observer;
} /// <summary>
/// 取消绑定订阅
/// </summary>
/// <param name="observer"></param>
public void RemoteObserver(NotifyEventHandler observer)
{
NotifyEvent -= observer;
} /// <summary>
/// 发布博客通知
/// </summary>
public void PublishBlog()
{
if(NotifyEvent != null)
{
NotifyEvent(this);
}
}
}
}
具体的博客类
/*****************************************************
* ProjectName: _11DesignPattern_ObserverDelegateTest
* Description:
* ClassName: JiYFBlog
* CLRVersion: 4.0.30319.42000
* Author: JiYF
* NameSpace: _11DesignPattern_ObserverDelegateTest
* MachineName: JIYF_PC
* CreateTime: 2017/5/23 20:55:58
* UpdatedTime: 2017/5/23 20:55:58
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_ObserverDelegateTest
{
/// <summary>
/// 具体的订阅博客类
/// </summary>
public class JiYFBlog : Blog
{
public JiYFBlog(string name, string blogTitile, string blogInfo)
: base(name, blogTitile, blogInfo)
{ }
}
}
订阅者类
/*****************************************************
* ProjectName: _11DesignPattern_ObserverDelegateTest
* Description:
* ClassName: Observer
* CLRVersion: 4.0.30319.42000
* Author: JiYF
* NameSpace: _11DesignPattern_ObserverDelegateTest
* MachineName: JIYF_PC
* CreateTime: 2017/5/23 20:56:50
* UpdatedTime: 2017/5/23 20:56:50
*****************************************************/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_ObserverDelegateTest
{
/// <summary>
/// 具体的订阅者类
/// </summary>
public class Observer
{
/// <summary>
/// 订阅者名字
/// </summary>
private string m_Name;
public string Name
{
get { return m_Name; }
set { m_Name = value; }
} /// <summary>
/// 订阅者构造函数
/// </summary>
/// <param name="name">订阅者名字</param>
public Observer(string name)
{
this.m_Name = name;
} /// <summary>
/// 订阅者接受函数
/// </summary>
/// <param name="blog"></param>
public void Receive(object obj)
{
Blog blog = obj as Blog;
if (blog != null)
{
Console.WriteLine("订阅者:\"{0}\"观察到了:{1}发布的一篇博客,标题为:{2},内容为:{3}", Name, blog.BlogName, blog.BlogTitle, blog.BlogInfo);
}
}
}
}
测试代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _11DesignPattern_ObserverDelegateTest
{
class Client
{
static void Main(string[] args)
{
Console.WriteLine("--全部订阅者--");
//创建一个JiYF的博客
Blog jiyfBlog = new Blog("JiYF笨小孩","丑小鸭","丑小鸭的故事"); //创建订阅者
Observer obsZhangsan = new Observer("张三");
Observer obsLiSi = new Observer("李四");
Observer obsWangwu = new Observer("王五"); //添加对JiYF博客的订阅者
jiyfBlog.AddObserver(new NotifyEventHandler(obsZhangsan.Receive));
jiyfBlog.AddObserver(new NotifyEventHandler(obsLiSi.Receive));
jiyfBlog.AddObserver(new NotifyEventHandler(obsWangwu.Receive)); ////通知订阅者
jiyfBlog.PublishBlog(); Console.WriteLine("---移除订阅者张三--");
jiyfBlog.RemoteObserver(new NotifyEventHandler(obsZhangsan.Receive));
jiyfBlog.PublishBlog();
Console.ReadLine();
}
}
}
运行结果:

源代码工程文件下载
C#设计模式--观察者模式(发布-订阅模式)的更多相关文章
- 设计模式 | 观察者模式/发布-订阅模式(observer/publish-subscribe)
定义: 定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态发生变化时,会通知所有观察者对象,使他们能够自动更新自己. 结构:(书中图,侵删) 一个抽象的观察者接口, ...
- [转] JavaScript设计模式之发布-订阅模式(观察者模式)-Part1
<JavaScript设计模式与开发实践>读书笔记. 发布-订阅模式又叫观察者模式,它定义了对象之间的一种一对多的依赖关系.当一个对象的状态发生改变时,所有依赖它的对象都将得到通知. 例如 ...
- C# 委托和事件 与 观察者模式(发布-订阅模式)讲解 by天命
使用面向对象的思想 用c#控制台代码模拟猫抓老鼠 我们先来分析一下猫抓老鼠的过程 1.猫叫了 2.所有老鼠听到叫声,知道是哪只猫来了 3.老鼠们逃跑,边逃边喊:"xx猫来了,快跑啊!我是老鼠 ...
- JavaScript设计模式(发布订阅模式)
发布—订阅模式又叫观察者模式,它定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知.在JavaScript开发中,我们一般用事件模型来替代传统的发布—订阅模式 ...
- SpringBoot事件监听机制及观察者模式/发布订阅模式
目录 本篇要点 什么是观察者模式? 发布订阅模式是什么? Spring事件监听机制概述 SpringBoot事件监听 定义注册事件 注解方式 @EventListener定义监听器 实现Applica ...
- javaScript设计模式:发布订阅模式
发布订阅模式的思想是在观察者模式的基础上演变而来,在观察者模式中客户端监听到对象某个行为就触发对应任务程序.而在发布订阅模式中依然基于这个核心思想,所以有时候也会将两者认为是同一种设计模式.它们的不同 ...
- js设计模式之发布/订阅模式模式
一.前言 发布订阅模式,基于一个主题/事件通道,希望接收通知的对象(称为subscriber)通过自定义事件订阅主题,被激活事件的对象(称为publisher)通过发布主题事件的方式被通知. 就和用户 ...
- 浅谈js设计模式之发布 — 订阅模式
发布 — 订阅模式又叫观察者模式,它定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知.在 JavaScript开发中,我们一般用事件模型来替代传统的发布 — ...
- Javascript设计模式之发布-订阅模式
简介 发布-订阅模式又叫做观察者模式,他定义了一种一对多的依赖关系,即当一个对象的状态发生改变的时候,所有依赖他的对象都会得到通知. 回忆曾经 作为一名前端开发人员,给DOM节点绑定事件可是再频繁不过 ...
- javascript中的设计模式之发布-订阅模式
一.定义 又叫观察者模式,他定义对象间的依照那个一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将的到通知.在javascript中,我们一般用时间模型来替代传统的发布-订阅模式 二 ...
随机推荐
- hdu4607Park Visit 树的直径
//给一棵双向树,数中边的权值为1,问对于这颗树中走k个节点的最短路径 //假设k小于这颗数的直径加1,那么走k个节点就没有反复的路,假设大于 //那么大于的节点都须要走两遍 #include< ...
- Java断言绝对不是鸡肋
在防御式编程中经常会用断言(Assertion)对参数和环境做出判断,避免程序因不当的输入或错误的环境而产生逻辑异常,断言在很多语言中都存在,C.C++.Python 都有不同的断言表示形式.在Jav ...
- Go语言之高级篇beego框架之请求数据处理
1.Controller中数据参数处理 获取参数:我们经常需要获取用户传递的数据,包括 Get.POST 等方式的请求,beego 里面会自动解析这些数据,你可以通过如下方式获取数据: GetStri ...
- STM32 F103 F407 F429 F767对比图
- 4种事务的隔离级别,InnoDB怎样巧妙实现?
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/z50L2O08e2u4afToR9A/article/details/82186189 事务ACID ...
- 关于Java变量的可见性问题
转自:http://www.importnew.com/19434.html 博文前提 最近在oschina问答板块看到了一个关于java变量在工作内存和主存中的可见性问题:synchorized,s ...
- 国内混合APP开发技术选型
http://www.sunzhongwei.com/weex-react-native-ionic-technology-selection 选谁? 企业级应用是要考虑性能和流畅度的, 如果只是做个 ...
- CentOS7下 让Docker pull命令使用squid做http代理拉取目标镜像仓库的镜像
场景,如下图所示: 服务器B具有两个网卡,分别和服务器A和服务器C互通,这里想要在服务器C上借助服务器B作为桥梁,拉取镜像仓库服务器A上的镜像. 思路也很简单,在服务器上搭建HTTP代理服务,服务器C ...
- 【asp.net core】Publish to a Linux-Ubuntu 14.04 Server Production Environment
Submary 又升级了,目录结构有变化了 . project.json and Visual Studio 2015 with .NET Core On March 7, 2017, the .NE ...
- 【SqlServer】SqlServer的常规操作
创建一张新表,不负责任何数据(该表不会有原来表的主键.索引等等) select * into NewTable from OldTable where 1<>1; 创建一张新表,并且复制旧 ...