EventBus 3.0

入门

简介

GitHub:EventBus

Android optimized event bus that simplifies communication between Activities, Fragments, Threads, Services, etc. Less code, better quality.

android最佳的事件总线,简化了Activities、Fragments、Threads、Services等之间的通信,代码更少,质量更好。

EventBus

EventBus是greenrobot团队Markus Junginger开源的发布-订阅事件总线库。主要用于:适合当做组件间的通讯工具使用,主要用来传递消息。请求网络,等网络返回时通过Handler或Broadcast通知UI,两个Fragment之间需要通过Listener通信,这些需求都可以通过EventBus实现。。。。

优点有

简洁性。简化了应用程序内各组件间、组件与后台线程间的通信。
开销小。代码更优雅。
低耦合。将发送者和接受者解耦。

缺点是

事件一般都要写个对应的 Event 类出来,导致会都很多这些类;
可读性降低。两个模块间交互,不如原来定位快。
EventBus3.0较之前版本:用注解的方式代替约定的方法名规范。

其他框架对比

RxBus:使用RxJava实现的库。缺点有:如何实现sticky功能;如何处理RxJava的订阅操作中的异常导致的订阅的中断(因为RxJava中出现异常会直接跳转OnError,事件流就被中断了);使用运行时(Run-time)注解的话,反射的性能问题。

Guava:Google的Guava是一个庞大的库,EventBus只是它附带的一个小功能,因此实际项目中使用并不多。

Otto:square的otto修改自 Guava。但已停止维护,临死前推荐大家使用RxBus。

基础用法

使用

基本使用大致分为5个步骤:

1,创建事件类;(用一个POJO封装传递数据。如果是简单的基本数据类型,不封装也行。)

2,在订阅事件处注册事件;(在发送端)

EventBus.getDefault().register(this);

3,发送事件;(此为发布者,在发送端)

EventBus.getDefault().post(new EventPOJO("Hello EventBus!"));

4,接收,处理事件;(此为订阅者,在接收端)

//接收方法可以随便起名,但需要添加Subscribe注解。
@Subscribe(threadMode= ThreadMode.MAIN)

public voidonEvent(EventPOJO messageEvent) {

    tv_message.setText(messageEvent.getMessage());

}

5,注销总线。

EventBus.getDefault().unregister(this);

@Subscribe

ThreadMode提供了四种类型。MainThread 主线程,BackgroundThread 后台线程,Async 后台线程,PostThread 发送线程(默认)。

  1. POSTING(默认):如果使用事件处理函数指定了线程模型为POSTING,那么该事件在哪个线程发布出来的,事件处理函数就会在这个线程中运行,也就是说发布事件和接收事件在同一个线程。在线程模型为POSTING的事件处理函数中尽量避免执行耗时操作,因为它会阻塞事件的传递,甚至有可能会引起ANR。
  2. MAIN:事件的处理会在UI线程中执行。事件处理时间不能太长,长了会ANR的。
  3. BACKGROUND:如果事件是在UI线程中发布出来的,那么该事件处理函数就会在新的线程中运行,如果事件本来就是子线程中发布出来的,那么该事件处理函数直接在发布事件的线程中执行。在此事件处理函数中禁止进行UI更新操作。
  4. ASYNC:无论事件在哪个线程发布,该事件处理函数都会在新建的子线程中执行,同样,此事件处理函数中禁止进行UI更新操作。

sticky机制

sticky机制俗称粘性事件。即发送事件之后再订阅该事件也能收到该事件。sticky默认值为false,在Subscribe方法中可以更改。

@Subscribe(threadMode = ThreadMode.POSTING,priority = 1, sticky = true)

priority 值表示优先级。值越小优先级越低,默认为0。

1,订阅粘性事件;

bt_subscription.setOnClickListener(newView.OnClickListener() {
    @OverridepublicvoidonClick(View v) {//注册事件
        EventBus.getDefault().register(MainActivity.this);
    }
});

2,处理粘性事件;

@Subscribe(threadMode = ThreadMode.POSTING,priority=0, sticky =true)
publicvoidononMoonStickyEvent(MessageEvent messageEvent){
    tv_message.setText(messageEvent.getMessage());
}

3,发送粘性事件。

bt_subscription.setOnClickListener(newView.OnClickListener() {
    @OverridepublicvoidonClick(View v) {
        EventBus.getDefault().postSticky(newMessageEvent("粘性事件"));
        finish();
    }
});

进阶用法

添加processor

由于使用了注解,性能会降低。使用processor在编译的时候为注册类构建了一个索引,而不是在运行时,这样的结果是其让EventBus 3.0的性能提升了一倍,相比2.4来说,其会是它的3到6倍。

EventBus性能对比

使用processor有两种方案。<一>添加依赖:provided'de.greenrobot:eventbus-annotation-processor:3.0.0-beta1';
<二>添加依赖:apt 'org.greenrobot:eventbus-annotation-processor:3.0.1'。在Application中添加EventBus.builder().addIndex(newMyEventBusIndex()).installDefaultEventBus();。

源码解析

EventBus是用了反射,并且维护了一个全局map,记录event到method的映射。

通过注解把类中的所有订阅者和订阅信息提取出来,这里有很多过滤条件保证提取的正确性。

EventBus中有两个映射表,subscriptionsByEventType中存放所有event和对应的订阅者,typesBySubscriber中存放每个订阅者对应的事件。提取出来的订阅者和信息就被存进了这两个表里。

private final Map<Class<?>, CopyOnWriteArrayList<Subscription>>

subscriptionsByEventType;

private final Map<Object, List<Class<?>>> typesBySubscriber;

所有的的poster被放在3个队列中,mainThreadPoster、backgroundPoster、asyncPoster。这3个队列是用链表实现的,让poster在各自的线程排队,等候处理。如果是POSTING类型的事件,就直接执行了,不用排队。

错误汇总

post了消息之后,你的订阅者有多个,每一个都接收吗?能否做到指定接收者。

使用经验

EventBus最好都定义在一个包下,分散开后期不好维护。

参考文章

Android事件总线(一)EventBus3.0用法全解析 刘望舒

EventBus使用详解

EventBus 3.0的用法详解(一)

EventBus3.0源码解析

老司机教你“飙”EventBus3

快速Android开发系列通信篇之EventBus

EventBus使用详解(一)——初步使用EventBus 讲的2.X,算的上比较早的blog了。贴上纪念下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351