EventBus & Otto的使用和比较

EventBus主要用来消息/事件的传递,却能实现组建之间的解耦。对比其他的消息传递:
  • ** 使用监听器接口(Listener Interface):**

1、一个实现了监听器接口的类,必须把它自身注册到它想要监听的类中去。这就使监听与被监听之间保持强关联关系,而且不利于单元测试。

2、对比:而EventBus则起到了桥梁作用,想要监听什么对象/事件,在EventBus中去注册(register(Object));想要发送某事件的消息,使用EventBus去post(post(Event))。这样便通过EventBus实现了监听者和被监听者的解耦。

  • ** 使用广播(BroadCastReciver): **

1、使用广播的代码臃肿(还有一种说法:它们内部的实现都需要IPC,单从传递效率上来讲,可能并不太适合上层的组件间通信),而且Intent传递数据时,在编译时并不能检查出所设的extra类型与收到时的类型一致。所以一个很常见的错误便是你或者你团队中的其他人改变了Intent所传递的数据,但忘记了对全部的接收器(receiver)进行更新。这种错误在编译时是无法被发现的,只有在运行时才会发现问题。

2、对比:使用EventBus所传递的消息/事件,是我们自己定义的Event类,由于接受方和发送方都是与这些类实例打交道,所以所有的数据都可以进行类检查,这样任何由于类型不一致导致的错误都可以在编译期被发现。

  • startActivityForResult()和onActivityResult()方法:

1、传统的Activity间的消息传递便是通过
startActivityForResult和onActivityResult,会产生较多的状态或逻辑判断,而且Intent或Bundle传值还得检测类型,容易发生错误。更复杂些的场景,比如A启动B,B启动C,在C中用户的操作需要更新A和B,那么判断逻辑就更麻烦了。

2、对比:使用EventBus可以很简单地解决以上问题,能够摆脱Activity间通信使用onActivityResult的方法,同样可以摆脱Fragment间通信通过宿主Activity使用Interface的方式,还有Fragment和Activity间通信使用Interface的方式。

EventBus可以用来处理异步并发消息。

Android中有很多执行异步操作的方法:AsyncTask、Loader、Executor等。

  • ** AsyncTask **
    AsyncTask被设计成为一个工具类,要求我们尽量执行一些短小的操作,如果需要在线程中执行较长时间的任务,那么建议直接使用java.util.concurrent包中提供的各种API,如Executor、 ThreadPoolExecutor以及FutureTask。

  • ** Loader **
    Android 3.0引入了Loader,来解决Activity/Fragment生命周期的问题,但Loader仍然要维持Activity/Fragment中的callback接口LoaderManager.LoaderCallbacks(其中包含onCreateLoader()、onLoadFinished()和onLoaderReset方法),在onCreateLoader()中要实现自己的DataLoader,可以继承抽象类AsyncTaskLoader<T>拓展泛型并覆写其loadInBackground()方法。

  • ** EventBus **
    EventBus中内置了并发处理机制,既支持工作线程向UI线程发送消息/事件,也支持从主线程发消息,工作线程来处理响应。

1、onEvent(T event)对应ThreadMode.PostThread。该方法的执行和事件发送者在同一个线程中,适用于对是否在主线程执行无要求的情况,但post线程为主线程,则不能有耗时操作。

2、onEventMainThread(T event)对应ThreadMode.MainThread。在主线程执行,不论事件从哪个线程发送过来。

3、onEventBackgroundThread(T event)对应ThreadMode.Background Thread如果发送事件的线程不是UI线程,则运行在该线程中。如果发送事件的是UI线程,则它运行在由EventBus维护的一个单独的线程中。多个事件会同步地被这个单独的后台线程所处理。适用于轻微耗时的操作,比如读写数据库。

4、onEventAsync(T event)对应ThreadMode.Async。运行在单独的工作线程中,不论发送事件的线程是否为主线程。跟BackgroundThread不一样,该模式的所有线程是独立的,因此适用于长耗时操作,例如网络访问。

这样通过EventBus就可以不用维护引用和回调接口,从而实现组件间的消息/事件传递。

EventBus处理网络请求并通知UI

因为EventBus提供了多种ThreadMode,我们完全也可以使用EventBus来处理耗时操作。比如在打开一个Activity时,我们暂称为“A”,它会从服务端获取数据并进行显示,我们首先注册EventBus;然后定义onEventMainThread(Response data)方法,接收请求到的数据并刷新界面;另外,还需要调用post(Request rq)来通知我们定义的RequestManger来访问网络并请求数据,当然RequestManger注册并定义了onEventAsync(Request rq)方法,我们在该方法中请求网络数据并最后调用EventBus.post(Response data)方法去通知“A”刷新UI。

EventBus默认创建了线程池为Executors.newCachedThreadPool(),Async和Background模式下都是使用该线程池执行任务,当然我们可以根据情况去定义自己的ExecutorService。

EventBus的一些局限(因为3.0版本有较大的更新,因此以下有些观点是陈旧的,已做更正)

1、EventBus是采用反射的方式对整个注册的类的所有方法进行扫描来完成注册,当然会有性能上的影响。[3.0中EventBus提供了EventBusAnnotationProcessor注解处理器来在编译期通过读取@Subscribe()注解并解析、处理其中所包含的信息,然后生成java类(默认EventBusIndex类在build文件夹中)来保存所有订阅者关于订阅的信息,这样就比在运行时使用反射来获得这些订阅者的信息速度要快]
2、EventBus必须定义以onEvent开头的几个方法,代码中语境比较突兀,且有可能会导致拼写错误。[EventBus 3.0支持注解方式,而且支持注解中设置ThreadMode、sticky事件、优先级],like this:

@Subscribe(threadMode = ThreadMode.BACKGROUND, sticky = true, priority = 100)
public void test(String str) { 
}

3、用EventBus来处理网络请求太过简单粗暴,考虑使用Volley处理网络请求,volley支持请求缓存、请求队列和请求优先级,代码的结构清晰、扩展性强,很适合客户端轻小的请求。关于Volley可以看我的 Volley Demo和源码注释

更好的解决方案(推荐使用EventBus 3.0)

1、Otto同样是事件总线框架(Otto Demo和源码注释),满足消息/事件传递的同时,也实现了组件间的解耦。跟EventBus一样,采用反射的方式对注册类中的所有方法进行扫描和调用。
2、Otto不同于EventBus的是,它使用注解的方式(@Subscribe、@Produce)来标注方法,这一点比EventBus优雅。
3、Otto没有像EventBus那样强大实现了4种ThreadMode,Otto中在接口ThreadEnforcer以及内部的实现域ANY和MAIN,在MAIN内部有一个是否是主线程的检查,而ANY不做任何检查的事情。因此Otto更多的使用场景是在主线程中,相对是轻量级的。
4、综合上面EventBus处理网络请求的缺陷,因此我觉得比较合理的方案是,使用Volley+Otto组合的方式来处理异步网络请求和UI更新,以及组件间的消息传递。
5、EventBus 3.0版本支持注解以及在编译期扫描事件订阅方法,性能上有较大提升,因此推荐使用EventBus 3.0版本。(这里有EventBus的官方性能测试对比报告:https://github.com/greenrobot/EventBus/blob/master/COMPARISON.md

更多可以参考我的:EventBus Demo和源码注释

转载请标明出处:http://www.jianshu.com/p/cb39a0018db1

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

推荐阅读更多精彩内容

  • 文章基于EventBus 3.0讲解。首先对于EventBus的使用上,大多数人还是比较熟悉的。如果你还每次烦于使...
    Hohohong阅读 2,281评论 0 6
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,870评论 25 707
  • 前言:EventBus出来已经有一段时间了,github上面也有很多开源项目中使用了EventBus。所以抽空学习...
    Kerry202阅读 1,282评论 1 2
  • 最近在项目中使用了EventBus(3.0),觉得非常好用,于是就看了一些关于EventBus源码分析的文章,现在...
    shenhuniurou阅读 1,494评论 0 4
  • 我不甘于旅行 所以戈壁滩上的那堆柴火 谁爱点燃谁去点 我在流动的金色沙粒中 凭心而论 若就这样睡着 总比死在大地上...
    在山阅读 410评论 0 0