再看C#中的委托和事件

         一口一个设计模式--观察者模式中,我们已经知道如何应用观察者模式,但通过最近的深入学习,发现观察者模式存在以下缺陷:

   1.抽象通知者依赖于抽象观察者;

   2.每个具体观察者被调用的方法必须一致。

   比如在机房收费系统中关于观察者模式的应用如下图所示:

         

  这个设计存在以下问题:

  1.抽象通知者需要把抽象观察者中的所有具体对象添加到集合中,以备向每个具体观察者发送通知消息;

  2.在发送通知消息时,具体观察者中被通知的方法必须一致,否则具体通知者无法完成遍历;

  3.更新卡内余额、更新学生上机状态和保存学生上机记录这三个类没什么相同的地方,抽象出来的抽象观察者有点牵强。

  经过苦苦的思索,我发现用C#中的委托和事件机制解决上述问题,代码结构如下:

  抽象通知者:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public interface Informer
    {
        void Notify();
        string Action { get; set; } 
    }
}
             具体通知者--正常下机:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public class Down : Informer
    {
        //声明一个委托
        public delegate void EventHandler();
        //为委托EventHandler添加事件Update
        public event EventHandler Update;

        public void Notify()
        {
            this.Update();
        }

        private string action;

        public string Action
        {
            get { return action; }
            set { action = value; }
        }
    }
}
           具体通知者--强制下机:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public class ForceDown : Informer
    {
        //声明一个委托
        public delegate void EventHandler();
        //为委托EventHandler添加事件Update
        public event EventHandler Update;

        public void Notify()
        {
            this.Update();
        }

        private string action;
        public string Action
        {
            get { return action; }
            set { action = value; }  
        }
    }
}
    这个我们不需要抽象观察者,直接具体观察者--更新卡内余额:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public class UpdateBalance
    {
        protected Informer dn;
        public UpdateBalance(Informer dn)
        {
            this.dn = dn;
        }
        public void ModifyBalance()
        {
            Console.WriteLine("{0}!该卡要下机,更新余额!", dn.Action);
        }
    }
}
            具体观察者--更新学生上机状态:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public class OffLine
    {
        protected Informer dn;
        public OffLine(Informer dn)
        {
            this.dn = dn;
        }
        public void DelOnlineInfo()
        {
            Console.WriteLine("{0}!该卡要下机,删除正在上机表中的该卡信息!", dn.Action);
        }
    }
}
            具体观察者--保存学生上机记录:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    public class SaveRec
    {
        protected Informer dn;
        public SaveRec(Informer dn)
        {
            this.dn = dn;
        }
        public void SaveLineRec()
        {
            Console.WriteLine("{0}!该卡要下机,保存上机记录信息!", dn.Action);
        }
    }
}
           main方法:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 观察者模式_委托
{
    class Program
    {
        static void Main(string[] args)
        {
            Down dn = new Down();

            OffLine ol = new OffLine(dn);
            UpdateBalance up = new UpdateBalance(dn);
            SaveRec sr = new SaveRec(dn);

           dn.Update+=new Down.EventHandler(ol.DelOnlineInfo);
           dn.Update += new Down.EventHandler(up.ModifyBalance);
           dn.Update += new Down.EventHandler(sr.SaveLineRec);

           dn.Action = "刷卡下机";
           dn.Notify();
        }
    }
}

           程序执行结果如下所示:

  

  通过执行结果,我们可以发现,在main方法中的正常下机事件dn.Update发生后,通过委托Down.EventHandler绑定到dn.Update事件上的三个方法ol.DelOnlineInfo、up.ModifyBalance、sr.SavelineRec都会被执行,达到观察者模式的效果。

  现在我们可以发现:抽象通知者不再依赖于抽象观察者,因为抽象通知者通过委托通知具体观察者;具体观察者中被调用的方法也没必要一致,因为只需把想要执行的方法绑定到事件上就行;具体通知者也不必存在共性,不必牵强的对这三个类进行抽象。委托+事件完美解决上述问题。  

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 18
    评论
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值