前言

EventBus是一个Android版本的页面间通信库,这个库让页面间的通信变得十分容易且大幅降低了页面之间的耦合。小弟之前玩Android的时候就用得十分顺手,现在玩uwp就觉得应该在这平台也写个类似的库。

这个库原理很简单,就是把观察者模式封装成库,页面想收到某类通知就注册相关事件,在其他页面发出通知后就做响应。

LLQNotifier的使用:

  1. //声明一种通知事件
  2. public class Event1
  3. {
  4. public string Flag { get; set; }
  5. }
  6. //注册并监听事件
  7. public class subscriber
  8. {
  9. public subscriber()
  10. {
  11. LLQNotifier.Default.Register(this);
  12. }
  13. //给收到通知后要回调的方法加上SubscriberCallback属性
  14. [SubscriberCallback(typeof(Event1), NotifyPriority.Lowest, ThreadMode.Background)]
  15. public void Test()
  16. {
  17. Debug.WriteLine("->>>>>>>>>>subscriber>>Test");
  18. }
  19. }
  20. //在某个地方发起事件通知
  21. LLQNotifier.Default.Notify(new Event1() { Flag = "flag" });

这样就可以了,事件接收者和发起事件通知的人互不可见,只要通知一声,所有注册这个事件的函数都会执行,并且可以设置执行的优先级,可以看到上面SubscriberCallback属性有三个参数:

  1. 事件类型,表明只有接到这种事件的通知才会响应。
  2. 优先级,总共有5级,实际运用中可能会碰到注册同一种事件并需要控制执行先后顺序,这时优先级就派上用场。
  3. 线程模式,有三种:主线程,当前线程和后台线程,比方说这个方法里做的是UI相关的就用ThreadMode.Main,如果是计算相关就可以用ThreadMode.Current或ThreadMode.Background。

在EventBus中还有个sticky的概念,粘性,事件在发起后一段时间,本来所有注册者都已经响应过了,这时再有其他注册者进来按道理应该是收不到这个事件通知了,不过有了这个sticky的话就可以让新进来的注册者也能响应到这个事件,不过在LLQNotifier里暂时还没有实现这个功能,个人觉得实用性不是太强。

LLQNotifier里的注册通知的具体实现:

  1. //注册事件,可以看到这个方法主要就是把注册者的响应方法和事件类型加到Dictionary里面,当然要先把注册里用Attribute标记的方法找出来并缓存
  2. public void Register(object subscriber)
  3. {
  4. if (_syncContext == null && SynchronizationContext.Current != null)
  5. _syncContext = SynchronizationContext.Current;
  6. List<Type> subscriptTypes = new List<Type>();
  7. if (_subscriberDictWithType.ContainsKey(subscriber))
  8. {
  9. return;
  10. }
  11. //这里把注册者里attrubute标记的方法找出来并封装在subscription里
  12. IList<Subscription> subscriptionList = SubscriptionHandler.CreateSubscription(subscriber);
  13. foreach (var subscription in subscriptionList)
  14. {
  15. var subscriptionsOfType = _subscriptionDictByType.GetOrAdd(subscription.EventType, new List<Subscription>());
  16. //这里用了一个锁数组,因为在注册时也有可能会在其他线程在通知事件来遍历,所以需要用锁来保证线程安全
  17. lock (_locksForSubscription.GetOrAdd(subscription.EventType, new object()))
  18. {
  19. subscriptionsOfType.Add(subscription);
  20. subscriptionsOfType.Sort();
  21. }
  22. if (!subscriptTypes.Contains(subscription.EventType))
  23. {
  24. subscriptTypes.Add(subscription.EventType);
  25. }
  26. }
  27. _subscriberDictWithType.Add(subscriber, subscriptTypes);
  28. }
  1. //接下来就是通知了
  2. public void Notify(object eventObj)
  3. {
  4. if (!_subscriptionDictByType.ContainsKey(eventObj.GetType()))
  5. return;
  6. var subscriptionsOfType = _subscriptionDictByType[eventObj.GetType()];
  7. List<Subscription> subList;
  8. lock (_locksForSubscription.GetOrAdd(eventObj.GetType(), new object()))
  9. {
  10. subList = subscriptionsOfType.ToList();
  11. }
  12. foreach (var subscription in subList)
  13. {
  14. if (subscription.IsSubscriberAlive && _subscriberDictWithType.ContainsKey(subscription.Subscriber))
  15. {
  16. DispatchNotification(subscription, eventObj);
  17. }
  18. }
  19. }
  1. //还有取消注册,适合用在Dispose时
  2. public void Unregister(object subscriber)
  3. {
  4. List<Type> types = null;
  5. if(_subscriberDictWithType.TryGetValue(subscriber, out types))
  6. {
  7. foreach(var type in types)
  8. {
  9. RemoveSubscription(type, subscriber);
  10. }
  11. }
  12. }

垃圾回收

因为注册时会把注册的对象保存起来,强引用的话会导致对象不能被GC回收,表现在应用里就是页面只要打开一次,内存就会被占用,即使页面已经关掉,内存不回收,这就是内存泄露了。

所以在Subscription里的_subscriber是作为了一个WeakReference存在,这就避免了subscriber不能回收的情况。不过还有一个问题,subscriber在有些地方是作为Key存在的,WeakReference作key的话,对象被回收了Dictionary不会做改变,这样可能导致Dictionary里垃圾越来越多,Value也不能被回收掉。好在有个专门处理这个问题的集合:ConditionalWeakTable,这个table可以在key被回收后删除这条记录,完美解决上面的问题。LLQNotifier里有两个地方用了到这个table,一个是Subscription里对method做的缓存,另一个是_subscriberDictWithType用来存储对象和对象包含的事件。

性能

关于性能我自己测试的结果是单个通知到响应时间小于1毫秒,10万个会在100毫秒以内,不过相信正常情况下是不会有这么多通知的,我在知乎日服读读日报里大量用到LLQNotifier,没发现任何性能上的影响。

开源

项目开源地址:https://github.com/brookshi/LLQNotifier,欢迎Fork/Star

UWP开源项目 LLQNotifier 页面间通信利器(移植EventBus)的更多相关文章

  1. [转] 微信小程序页面间通信的5种方式

    微信小程序页面间通的5种方式 PageModel(页面模型)对小程序而言是很重要的一个概念,从app.json中也可以看到,小程序就是由一个个页面组成的. 如上图,这是一个常见结构的小程序:首页是一个 ...

  2. JavaScript之iframe页面间通信

    [1] iframe父子页面间通信 1.相互调用对方的方法 |> 子级页面调用父级页面 window.parent.父级页面方法(args) |> 父级页面调用子级页面 document. ...

  3. 如何实现electron多页面间通信

    如何实现electron多页面间通信 1,业务需求: 总共有两个页面,页面A显示数据,页面B处理数据,主线程Main 2,实现的技术方案: 在主线程中打开页面A和B,B页面不进行显示,主要负责处理从A ...

  4. 【开源项目11】组件间通信利器EventBus

    概述及基本概念 **EventBus**是一个Android端优化的publish/subscribe消息总线,简化了应用程序内各组件间.组件与后台线程间 的通信.比如请求网络,等网络返回时通过Han ...

  5. js页面间通信方法(storage事件)(浏览器页面间通信方法)

    在写页面的时候有时会遇到这样的需求,需要两个页面之间传递数据或者一个事件.这个时候,就需要用到我今天所要讲的storage事件,学习这个事件之前,需要先了解localStorage的用法.具体用法可以 ...

  6. EventBus完全解析--组件/线程间通信利器

    github地址:https://github.com/greenrobot/EventBus 1, Android EventBus实战, 没听过你就out了 2,  Android EventBu ...

  7. React Native 中 跨页面间通信解决方案之 react-native-event-bus

    https://github.com/crazycodeboy/react-native-event-bus 用法: A页面和B页面中都有相同的列表,点击B页面中的收藏按钮,A页面会跟着更新 impo ...

  8. 从UWP到SWIFT-页面间反向传值

    页面1跳转到页面2,在页面2点击button后,页面1的内容被改变.实际使用 protocol(就是c#中的interface),将页面1的viewcontroller转换为protocol传入页面2 ...

  9. vue项目各页面间的传值

    githut地址:https://github.com/liguoyong/vueobj1 一.父子之间主键传值:(主要是在父主件里的子主件传递参数,然后再子主件里用props接收) 例如Father ...

随机推荐

  1. SQL优化技巧--远程连接对象引起的CTE性能问题

    背景 最近SSIS的开发过程中遇到几个问题.其中使用CTE时,遇到一个远程连接对象,结果导致严重的性能问题,为了应急我就修改了代码. 之前我写了一篇介绍CTE的随笔包含了CTE的用法等: http:/ ...

  2. centos7系统下安装nodejs开发环境

    1)安装基础工具(if not exists) yum install -y net telnet tools vim wget ntp 2)同步系统时间(if necessary) ntpdate ...

  3. WIN 下的超动态菜单(一)

    WIN 下的超动态菜单(一)介绍 WIN 下的超动态菜单(二)用法 WIN 下的超动态菜单(三)代码 作者:黄山松,发表于博客园:http://www.cnblogs.com/tomview/     ...

  4. Android中在sdcard上创建文件夹

    //在SD卡上创建一个文件夹    public void createSDCardDir(){     if(Environment.MEDIA_MOUNTED.equals(Environment ...

  5. C#中标准Dispose模式的实现与使用(条目17 实现标准的销毁模式)

    实现了Dispose模式与实现了IDisposable接口的区别就是:IDisposable的实现的可靠性(释放相关资源)要靠编程人员来解决(你确信你从来都一直调用了Dispose(Close)方法吗 ...

  6. Hibernate4中使用getCurrentSession报Could not obtain transaction-synchronized Session for current thread

    架个spring4+hibernate4的demo,dao层直接注入的sessionFactory,然后用getCurrentSession方法获取session,然后问题来了,直接报错: Could ...

  7. NOIP2010提高组 机器翻译 -SilverN

    /**/ #include<iostream> #include<cstdio> #include<cmath> #include<cstring> # ...

  8. codeforces118D. Caesar's Legions

    地址:http://www.codeforces.com/problemset/problem/118/D 题目: Gaius Julius Caesar, a famous general, lov ...

  9. [No000005]C#注册表操作,创建,删除,修改,判断节点是否存在

    //用.NET下托管语言C#操作注册表,主要内容包括:注册表项的创建,打开与删除.键值的创建(设置值.修改),读取和删除.判断注册表项是否存在.判断键值是否存在. //准备工作: //1:要操作注册表 ...

  10. poj 2481

    Cows Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 16163   Accepted: 5380 Description ...