由"猫,老鼠和主人"引出的委托,事件及观察者模型问题

时间:2022-06-01 23:30:17

这是一道非常经典的面试试题.在网上讨论的帖子很多,一些网友给出了十分精彩的解答.这里也只是其中一个比较精简的答案而已.虽然精简,但却通过简单的代码反映了许多人一直不是很清晰的委托,特别是事件的定义和注册问题,也简单涉及了观察者模型的问题.

using System;

namespace CatRatAndHost//猫,老鼠和主人,有趣的委托,事件及观察者模型问题
{
    class Program
    {
        static void Main(string[] args)
        {
            catclass cat = new catclass();
            manclass man = new manclass();
            for (int i = 0; i < 5; i++)
            {
                ratclass rat = new ratclass();//老鼠是观察者,它对猫的动作感兴趣.注意:在这个循环中,new了5只同名的rat.变量rat的作用范围只在{...}内有效.
                rat.subject = cat;//设置猫是观察对象,并向猫注册了猫叫事件的事件处理方法--老鼠逃跑.
                man.subject = rat;//而老鼠在这里又是观察对象,主人对它的跑感兴趣,所以向老鼠注册它的跑事件的事件处理方法.               
            }
            cat.catEvent += new catclass.catEventHandler(man.wake1);//直接向猫注册事件处理方法.
            man.IsWake = false;//主人入睡
            cat.catMiao();//猫叫,引起一连串的动作.
            Console.ReadKey();
        }
    }
    public interface IObserver //作为观察者,有一个设置观察对象的属性.
    {
        ISubject subject
        {
            set;
        }
    }
    public interface ISubject
    {}
    public class catclass : ISubject //猫对象,继承了观察对象的接口
    {
        public delegate void catEventHandler(object source, EventArgs e); //申明事件的委托
        public event catEventHandler catEvent; //定义事件

        public virtual void catMiao() //猫叫,并触发事件
        {
            Console.WriteLine("The cat: Miao...");
            if (catEvent != null)
            {
                catEvent(this, EventArgs.Empty);
            }
        }
    }
    public class ratclass : IObserver,ISubject //老鼠对象,继承观察者接口,同时也是观察对象.
    {
        public event EventHandler ratEvent; //利用.Net预定义的EventHandler委托直接定义事件.

        catclass cat = null; //注册事件处理方法时用到的被观察对象
        public ISubject subject //观察对象属性
        {
            set
            {
                cat = (catclass)value;
                cat.catEvent += new catclass.catEventHandler(run); //注册事件处理方法
            }
        }
        public virtual void run(object source, EventArgs e)//定义事件处理方法
        {
            Console.WriteLine("The Rat: Run...Run...Run...");
            if(ratEvent!=null)
            {
                ratEvent(this, EventArgs.Empty);
            }
        }
    }
    public class manclass : IObserver //主人同老鼠,其实主人也可能被猫叫惊醒.
    {
        private bool _iswake = true;
        public bool IsWake
        {
            get
            {
                return _iswake;
            }
            set
            {
                _iswake = value;
            }
        }

        ratclass rat = null;
        public ISubject subject
        {
            set
            {
                rat = (ratclass)value;
                rat.ratEvent += new EventHandler(wake2);
            }
        }
        public virtual void wake1(object source, EventArgs e)
        {
            if (!_iswake) //确保主人不会被重复惊醒
            {
                Console.WriteLine("The host: Waked by cat...");
                _iswake = true;
            }
        }
        public virtual void wake2(object source, EventArgs e)
        {
            if (!_iswake) //确保主人不会被重复惊醒
            {
                Console.WriteLine("The host: Waked by rat...");
                _iswake = true;
            }
        }
    }
}

代码的用到了事件的两种定义方法:

public event EventHandler ratEvent; //利用.Net预定义的EventHandler委托直接定义事件. 这种方法虽然简单,但对于事件的参数和返回值也都是只能按照预定义好的方法进行.但是,一般情况下,这种方式已足够了.
public delegate void catEventHandler(object source, EventArgs e); //申明事件的委托
public event catEventHandler catEvent; //定义事件,这种方式需要先申明一个相关的委托.然后通过new这个委托来完成事件的注册.相比前面的方式而言要麻烦一些,但是通过这种方式,我们可以自己定义参数列表和返回值,相对较灵活和更强大一些.

代码的用到了事件的两种注册方法:

其实只能算是一种方法,只是通过不能的途径来完成的而已,最终都是用"观察对象.事件名+=new 事件对应委托(观察者的事件处理方法)"公式来完成的.
一种是采用观察者的属性--观察对象来给观察对象注册事件处理方法.
如:rat.subject = cat;
别一种是申明完观察者后和观察对象,直接注册.
如:cat.catEvent += new catclass.catEventHandler(man.wake1);