ABP之事件总线(4)

在上一篇的随笔中,我们已经初步完成了EventBus,但是EventBus中还有诸多的问题存在,那么到底有什么问题呢,接下来我们需要看一看ABP中的源码是如何定义EventBus的。

1.第一个点

 

在ABP中提供了对Action类型的支持,而我们的自己定义的类中只是针对继承了IEventHandler的接口的类

2.第二个点

在ABP中使用了线程安全的ConcurrentDictionary来存放映射关系,因为EventBus作为一个单例存在,这是必须要考虑的。ConcurrentDictionary虽然保证了字典的线程安全,但是并不能保证List的线程安全,可能存在同一时间同时插入相同的Handler,所以在上面的方法中使用了加锁的方式。

3.第三个点

 

 从上面的源码中,我们可以看出EventBus使用了反射和容器(CastleWindsor)注入的方式触发,其实要实现完全解耦,实现灵活可配置,反射是一项必不可少的技术,但是凡事都都是讲究一个度,我们知道反射的缺点就是性能消耗比较大,随着反射技术的发展,反射的速度已经提升了许多,但是相对于直接调用,那还是存在一定的差距。但是只要讲究合理度,损耗一部分性能实现良好的系统,这是明智的。在上一篇随笔中,我们的所有的类型包括EventData和Eventhandler全部是使用的是反射的方式实现的,可想而知,这是消耗性能最大的,而在ABP中采用了加入的容器的方式,这将改善性能问题。其实还是那句话,容器这种接触耦合的方式还是离不开反射技术在里面。在Castle Windsor的源码中,它同样是维护了一个字典,只是将部分类通过反射的方式得到,来优化完全使用反射获取对象的方式。

4.第四个点

加入了异步的方式触发事件,通过重新开启一个线程的方式,提高执行的速度。

 首先来完善第一个问题,其实在事件总线最开始的时候我们已经尝试定义了一个统一的Handler只不过那个Handler定义的有点过头了。。。。,这个我们可以参照一下ABP的源码中ActionEventHandler中的定义

  internal class ActionEventHandler<TEventData> :IEventHandler<TEventData> where TEventData:EventData
    {
        /// <summary>
        /// Action委托具有一个TEventData类型的参数
        /// </summary>
        public Action<TEventData> Action { get; private set; }
        /// <summary>
        /// 因为就是为了统一创建Handler所以处理逻辑是通过构造函数传入而不是直接在Handle()方法中写死
        /// </summary>
        /// <param name="handler"></param>
        public ActionEventHandler(Action<TEventData> handler)
        {
            Action = handler;
        }
        public void Handle(TEventData evetData)
        {
            Action(evetData);
        }
    }

 我们在上一篇定义的Mouse类中,将我们的ActionHandler触发,简单的测试一下

 

 public void Come()
        {
            new ActionEventHandler<MouseEventData>((data)=>Console.WriteLine(data.Name)).Handle(new MouseEventData() {  Name="小黄"});
            //MouseEventHandler(new MouseEventData() { Name=this.Name});
            //EventBus.Default.Trigger(new MouseEventData() { Name=this.Name});
        }

 

第二个问题:解决并发性的问题此时需要用到锁,所以只需要在可能产生并发的代码中加入锁即可

  //尽量保证锁中的代码少
  public static object  lockObj=new object();
  public void Register(Type dataType, Type handlerType)
        {
            lock (lockObj)
            {
                if (mapDic.Keys.Contains(dataType))
                {
                    if (!mapDic[dataType].Contains(handlerType))
                    {
                        mapDic[dataType].Add(handlerType);
                    }
                }
                else
                {
                    mapDic[dataType] = new List<Type>() { handlerType };
                }
            }
        }
 public void Unregister(Type eventType, Type handler)
        {
            lock (lockObj)
            {
                if (mapDic.Keys.Contains(eventType))
                {
                    if (mapDic[eventType].Contains(handler))
                    {
                        mapDic[eventType].Remove(handler);
                    }
                }
            }
        }

 

第四个问题:加入异步的处理方法,其实就是单独开启一个线程执行需要同步执行的代码而已

 public Task TriggerAsync<TEventData>(TEventData eventData) where TEventData : IEventData
        {
            return Task.Run(() => Trigger(eventData));
        }
//调用一下
  static void Main(string[] args)
        {
          
            EventBus.Default.Register(typeof(MouseEventData), typeof(CatchEventHandler));
            Mouse m = new Mouse("老鼠1号");
            EventBus.Default.TriggerAsync(new MouseEventData() { Name = "xiaohuang" });
            m.Come();
            Console.Read();
        }

 第三个问题,涉及到容器,这一部分内容较多,将在接下来的随笔中学习

posted @ 2018-04-17 12:30  善良的小赵  阅读(1659)  评论(1编辑  收藏  举报