打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
c#设计模式

Observer 与 Subject 互为耦合,但是这种耦合的双方都依赖于抽象,而不依赖于具体

一、观察者模式

目的 概述 原理

二、   C#中的观察者模式

概述 模型与观察者基类 优点

三、   事例

题目:猫大叫,两只老鼠开始逃跑,主人醒来,宝宝也醒来了并且哭了起来

解决方案: 1.    建立模型(目标基类) 2.    建立观察者基类(单行为,多行为) 3.    建立具体目标 4.    建立具体观察者 5.    运行测试

 

一、      观察者模式

目的

我们都知道解决一个问题有N种解决方式,但在面向对象的设计中如何能做到“高内聚,低耦合”,设计可重用的对象才是我们追求的。在设计过程中,我们经常会接触到一种情况:一个对象的行为引发其它多个对象相应的行为。这时我们便可以通过观察者模式的设计思想来设计对象模型。

概述

观察者模式(Observer Pattern)是设计模式中行为模式的一种,它解决了上述具有一对多依赖关系的对象的重用问题。此模式的参与者分为两大类,一类是被观察的目标,另一类是观察该目标的观察者们。正因为该模式是基于“一对多”的关系,所以该模式一般是应用于由一个目标对象和N个观察者对象组成(当然也可以扩展为有多个目标对象,但我们现在只讨论前者)的场合。当目标对象的状态发生改变或做出某种行为时,正在观察该目标对象的观察者们将自动地、连锁地作出相应的响应行为。

原理

我们可以把观察目标理解为主动方、发布方、主体等;把观察者理解为被动方、订阅方、观察器等。目标是整个行为链的源头,其它观察者都依赖于它的变化而作出响应。为了实现低耦合,我们不能使用“直接调用”的方式而需要利用“订阅(清单)-通知”的机制去完成设计。通俗地说就是观察者向目标“订阅”它的改变,而目标发生改变后就“通知”所有已经“订阅”了它的改变的观察者,从而执行“订阅”的内容。这种机制的好处在于降低耦合度,分工明确,目标只负责在自身状态发生改变或做出某种行为时向自身的订阅清单发出“通知”,而不是直接调用观察者的行为(方法);观察者只负责向目标“订阅”它的变化,以及定义自身在收到目标“通知”后所需要做出的具体行为(也就是订阅的内容)。就像我们向出版社订阅报刊一样,出版社有新一期报刊发行时并不是直接跟每位订阅者联系,而是“通知”订阅者名单按顺序给每位订阅者发送所订报刊。

 

二、      C#中的观察者模式

概述

每种编程架构及程序语言,对观察者模式都有不通的具体实现。在.NET框架中,C#语言使用委托以及事件,可以很好的实现观察者模式。委托相当于“订阅清单”的角色,当目标中关联了该委托的事件被触发时,则委托将自动按序执行观察者注册于委托中的方法。

模型与观察者基类

我们把观察者模式的参与者都描述为派生自模型及观察者二个抽象基类的类。模型规划了事件,而观察者则规划了订阅及行为。

模型需要做的只是声明委托以及声明委托类型的事件。当然,还可以附加上封装了触发委托事件的方法。所有派生自模型的类都是具体目标,它们所要做的只是在适当的场合触发事件。(即发出“通知”)。

在观察者基类中,我们通过构造器将抽象的响应方法注册(订阅)于委托事件中。所有派生自观察者基类的类都是具体观察者。因为订阅行为已经在抽象基类完成,具体观察者需要做的只是通过覆盖观察者基类的方法去定义具体需要响应的行为,和通过构造器把需要观察的具体目标传递给基类构造器。

优点

通过对模型与观察者基类的分析可知,委托与事件的机制几乎消除了这两个模块之间的耦合,灵活性提高了很多。如果需要增加观察者,则只需要覆盖基类抽象方法及把观察目标传递给基类。

 

三、      事例

题目:猫大叫,两只老鼠开始逃跑,主人醒来,宝宝也醒来了并且哭了起来.

解决方案:

1.   建立模型(目标基类)

 1using System; 2 3  4 5namespace DelegateEvent 6 7{ 8 9    /**//// <summary>1011    ///     在Observer Pattern(观察者模式)中,此类作为所有Subject(目标)的抽象基类1213    /// 所有要充当Subject的类(在此事例中为"猫")都继承于此类.1415    ///     我们说此类作为模型,用于规划目标(即发布方)所产生的事件,及提供触发1617    /// 事件的方法.1819    ///     此抽象类无抽象方法,主要是为了不能实例化该类对象,确保模式完整性.2021    ///     具体实施:2223    ///     1.声明委托2425    ///     2.声明委托类型事件2627    ///     3.提供触发事件的方法2829    /// </summary>3031    public abstract class ModelBase3233    {3435        public ModelBase()3637        {3839        }4041        /**//// <summary>4243        /// 声明一个委托,用于代理一系列"无返回"及"不带参"的自定义方法4445        /// </summary>4647        public delegate void SubEventHandler(); 4849        /**//// <summary>5051        /// 声明一个绑定于上行所定义的委托的事件5253        /// </summary>5455        public event SubEventHandler SubEvent;5657 5859        /**//// <summary>6061        /// 封装了触发事件的方法6263        /// 主要为了规范化及安全性,除观察者基类外,其派生类不直接触发委托事件6465        /// </summary>6667        protected void Notify()6869        {7071            //提高执行效率及安全性7273            if(this.SubEvent!=null)7475                this.SubEvent();7677                7879        }8081    }8283}

 

 

2.   建立观察者基类(单行为,多行为)

//--------------------单行为---------------------

 

1using System; 2 3  4 5namespace DelegateEvent 6 7{ 8 9    /**//// <summary>1011    ///     在Observer Pattern(观察者模式)中,此类作为所有Observer(观察者)的抽象基类1213    /// 所有要充当观察者的类(在此事例中为"老鼠"和"人")都继承于此类.1415    ///     我们说此类作为观察者基类,用于规划所有观察者(即订阅方)订阅行为.1617    ///     在此事例中,规划了针对目标基类(ModelBase)中声明的"无参无返回"委托的一个1819    /// 方法(Response),并于构造该观察者时将其注册于具体目标(参数传递)的委托事件中.2021    ///     具体实施过程:2223    ///     1.指定观察者所观察的对象(即发布方).(通过构造器传递)2425    ///     2.规划观察者自身需要作出响应方法列表2627    ///     3.注册需要委托执行的方法.(通过构造器实现)2829    /// </summary>3031    public abstract class Observer3233    {3435        /**//// <summary>3637        /// 构造时通过传入模型对象,把观察者与模型关联,并完成订阅.3839        /// 在此确定需要观察的模型对象.4041        /// </summary>4243        /// <param name="childModel">需要观察的对象</param>4445        public Observer(ModelBase childModel)4647        {4849            //订阅5051            //把观察者行为(这里是Response)注册于委托事件5253            childModel.SubEvent+=new ModelBase.SubEventHandler(Response);5455        }5657 5859        /**//// <summary>6061        /// 规划了观察者的一种行为(方法),所有派生于该观察者基类的具体观察者都6263        /// 通过覆盖该方法来实现作出响应的行为.6465        /// </summary>6667        public abstract void Response();6869    }7071}7273

 

//-------------------多行为-------------------

 

 1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    /// 定义了另一个观察者基类.该观察者类型拥有两个响应行为.1011    /// 并在构造时将响应行为注册于委托事件.1213    /// (具体描述请参照另一观察者基类Observer)1415    /// </summary>1617    public abstract class Observer21819    {2021        /**//// <summary>2223        /// 构造时通过传入模型对象,把观察者与模型关联,并完成订阅.2425        /// 在此确定需要观察的模型对象.2627        /// </summary>2829        /// <param name="childModel">需要观察的对象</param>3031        public Observer2(ModelBase childModel)3233        {3435            //订阅3637            //把观察者行为(这里是Response和Response2)注册于委托事件3839            childModel.SubEvent+=new ModelBase.SubEventHandler(Response);4041            childModel.SubEvent+=new ModelBase.SubEventHandler(Response2);4243            4445        }4647        /**//// <summary>4849        /// 规划了观察者的二种行为(方法),所有派生于该观察者基类的具体观察者都5051        /// 通过覆盖该方法来实现作出响应的行为.5253        /// </summary>5455        public abstract void Response();5657        public abstract void Response2();5859    }6061}

 

3.  

 

建立具体目标

  

1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    ///     此类为观察者模式中的具体目标(即具体发布方),其继承于模型.1011    /// 其中包含(调用)了在模型中被封装好的触发委托事件的方法.1213    /// </summary>1415    public class Cat:ModelBase1617    {1819        public Cat()2021        {2223        }2425        /**//// <summary>2627        /// 定义了猫的一种行为----大叫2829        /// </summary>3031        public void Cry()3233        {3435            System.Console.WriteLine("Cat Cry..");3637            //调用了触发委托事件的方法.3839            //通知委托开始执行观察者已订阅的方法.4041            this.Notify();  4243         }4445    }4647}48

 

4.   建立具体观察者

//--------------具体观察者(老鼠)-------------

 

 1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    ///     此类为观察者模式中的具体观察者(即具体发布方),其继承于观察者基类.1011    /// 其中覆盖了观察者基类规划好的方法,实现了响应的具体行为.1213    /// </summary>1415    public class Mouse:Observer1617    {1819        /**//// <summary>2021        /// 观察者可以拥有自己的成员(字段或者方法).2223        /// 在此事例中增加了"老鼠的名字"2425        /// </summary>2627        private string name;2829        /**//// <summary>3031        ///     构造时确定观察者所需要观察的对象(具体目标),并传递给观察者基类构造器,3233        /// 实现响应行为(方法)的订阅.另外,为观察者实例初始化成员.3435        /// </summary>3637        /// <param name="name">老鼠的名字</param>3839        /// <param name="childModel">4041        ///     需要观察的对象(发布方).4243        ///     此处用模型基类来传递,是为了兼容所有派生于此模型的观察者,从而提高扩展性.4445        /// </param>4647        public Mouse(string name,ModelBase childModel):base(childModel)4849        {5051            //初始化字段(老鼠的名字)5253            this.name=name;         5455        }5657        /**//// <summary>5859        /// 覆盖了该类观察者需要作出的具体响应行为.6061        /// 此行为已在观察者基类中注册于委托事件,由委托事件调度执行,不需要直接调用.6263        /// </summary>6465        public override void Response()6667        {6869            //具体响应内容7071            System.Console.WriteLine(this.name+"开始逃跑");7273        }7475 7677    }7879}8081

 

//----------------具体观察者(主人)----------------

 

 1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    ///     此类为观察者模式中的具体观察者(即具体发布方),其继承于观察者基类.1011    /// 其中覆盖了观察者基类规划好的方法,实现了响应的具体行为.1213    /// </summary>1415    public class Master:Observer1617    {1819        /**//// <summary>2021        ///     构造时确定观察者所需要观察的对象(具体目标),并传递给观察者基类构造器,2223        /// 实现响应行为(方法)的订阅.2425        /// </summary>2627        public Master(ModelBase childModel):base(childModel)2829        {3031        }3233 3435        /**//// <summary>3637        /// 覆盖了该类观察者需要作出的具体响应行为.3839        /// 此行为已在观察者基类中注册于委托事件,由委托事件调度执行,不需要直接调用.4041        /// </summary>4243        public override void Response()4445        {4647            System.Console.WriteLine("主人醒来");4849        }5051    }5253}5455

 

//-------------------具体观察者(宝宝)-------------

 

 1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    ///     此类为观察者模式中的具体观察者(即具体发布方),其继承了订阅了2个响应行为的1011    /// 观察者基类.1213    ///     其中覆盖了观察者基类规划好的二个方法,实现了响应的具体行为.1415    /// </summary>1617    public class Master2:Observer21819    {2021        /**//// <summary>2223        ///     构造时确定观察者所需要观察的对象(具体目标),并传递给观察者基类构造器,2425        /// 实现响应行为(方法)的订阅.2627        /// </summary>2829        public Master2(ModelBase childBase):base(childBase)3031        {3233        }3435 3637        /**//// <summary>3839        /// 覆盖了该类观察者需要作出的具体响应行为.4041        /// 此行为已在观察者基类中注册于委托事件,由委托事件调度执行,不需要直接调用.4243        /// </summary>4445        public override void Response()4647        {4849            Console.WriteLine("baby醒来。。。。");5051 5253        }5455        /**//// <summary>5657        /// 覆盖了该类观察者需要作出的另一个响应行为.5859        /// </summary>6061        public override void Response2()6263        {6465            Console.WriteLine("开始哭闹。。。。。");6667        }6869    }7071}7273

 

5.   运行测试

 

1using System; 2 3 namespace DelegateEvent 4 5{ 6 7    /**//// <summary> 8 9    /// Observer Pattern(观察者模式)事例分析1011    /// 1213    /// 题目:猫大叫,两只老鼠开始逃跑,主人醒来,宝宝也醒来了并且哭了起来.1415    /// 1617    /// 关于目标(发布方):1819    ///     在此事例中,只有一个目标对象(发布方)猫,因为其他全部实体的行为都是2021    /// 响应它的"大叫"所执行的.猫是主动方,它的大叫引起一系列的连锁反应.2223    /// 2425    /// 关于观察者(订阅方):2627    ///     至于此事例的中的观察者分别有两大类,一类是听到猫大叫后只作出一种2829    /// 反应的观察者(老鼠,主人),另一类是听到锚大叫后会作出两种响应的观察者(3031    /// 宝宝).所以观察者分别需要派生于两个不同的观察者基类.3233    /// </summary>3435    public class SubMain3637    {3839        public SubMain()4041        {4243            4445        }4647        public static void Main()4849        {5051            //声明并实例化一个目标(即发布方)对象----猫5253            Cat myCat=new Cat();5455            //声明并实例化一个Mouse类型的观察者对象--名叫mouse1的老鼠.并把那只猫作为它所要观察的对象.5657            Mouse myMouse1=new Mouse("mouse1",myCat);5859            //类似地生成另一只名叫mouse2的老鼠(观察者),把同一只猫作为它的观察的对象.6061            Mouse myMouse2=new Mouse("mouse2",myCat);6263            //声明并实例化一个Master类型的观察者--主人,并同时把那只猫也作为他的观察对象.6465            Master myMaster=new Master(myCat);6667            //声明并实例化一个Master2类型的观察者--宝宝,同时把那只猫也6869            Master2 myLittleMaster=new Master2(myCat);7071 7273            //猫大叫,并触发了委托事件,从而开始按顺序调用观察者已订阅的方法.7475            myCat.Cry();7677 7879            Console.Read();8081        }8283    }8485}86

 参考:http://www.cnblogs.com/zhenyulu/articles/73723.html

本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
设计模式---“观察者模式"
一步步编写PHP的Framework(十八)
系统架构师谈企业应用架构之系统设计规范与原则2
C#设计模式-装饰器模式(Decorator Pattern)
抽象
批量删除C#注释(适用于vs开发环境)
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服