移动架构<第六篇>:EventBus事件总线框架

在Android的实际开发中,消息的通信是非常频繁的。结合Android的基础知识,常用的通信方式有Intent、Handler、Broadcast,它们都可以在Activity、Fragment、Service之间相互通信,但是呢?从架构的思维考虑,Android本身的通信方式会增加代码架构的复杂度。为了解决这个问题,必须要重新定制一个新的通信方案。
通信方式有两种:线程间通信跨进程通信
本文主要讲解EventBus框架的使用,它只能实现线程间的通信,不支持跨进程通信。

[github地址]

首先,EventBus对应的github地址需要记住,如下:

https://github.com/greenrobot/EventBus

EventBus的版本会被更新的,所以,想要知道EventBus的最新版本,需要打开对应的github地址查看。

[简单演示]

首先贴一下演示代码:

public class MainActivity extends AppCompatActivity {

    private Button button;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        //注册
        EventBus.getDefault().register(this);

        button = findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        //发送消息
                        EventBus.getDefault().post(new MessageEvent(100));
                    }
                }).start();
            }
        });
    }

    /**
     * 接收消息
     * @param event
     */
    @Subscribe(threadMode = ThreadMode.MAIN)
    public void onMessageEvent(MessageEvent event) {
        /* Do something */
        Log.d("yunchong", "MessageEvent的flag值为:"+event.getFlag());
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        //反注册
        EventBus.getDefault().unregister(this);
    }
}

EventBus的使用非常简单,如上代码所示,在onCreate方法里面注册,在onDestroy方法里面解注册,代码EventBus.getDefault().post发送消息,以及onMessageEvent方法接收消息。这种写法类似于Android的BroadCast,但和Android的BroadCast有本质的区别。

在一些项目中,我经常看到Activity和Fragment之间的通信使用广播来实现,我对广播的理解分为以下几点:

  • 常用的广播分为Broadcast(常规广播)和LocalBroadcast(本地广播)。前者发送的消息,整个系统都能收到,后者发送的消息只能当前应用可以收到。但是,由于大型项目中可能存在无数消息的传递,这样会导致广播的泛滥,可读性变差,如果非要使用广播的话建议使用后者;
  • 广播的注册方式有两种:动态注册和静态注册,本人不推荐静态注册方式,因为静态注册方式使广播常驻内存,广播是非常消耗内存的,静态注册方式不可取;
  • Android的广播机制不仅可以跨线程也能跨进程,但是广播的代码实在不能用优雅这个词来形容,从后期项目的维护成本出发,广播的代码还是比较紊乱的,加大了项目的维护成本。

所以,EventBus完全可以替代Android的广播,不仅如此,任何跨线程的通信都可以使用EventBus。那么,跨进程怎么玩?Android的跨进程常常使用AIDL机制,但考虑到代码的简洁性,推荐使用Hermes框架,Hermes框架不是本章的重点,所以把思路重新回到EventBus上。

要使用EventBus,一般需要注册,register方法的参数传递一个对象,大部分情况下,这个对象是当前Activity对象。

    //注册
    EventBus.getDefault().register(this);

为了防止内存泄漏,对应的还有反注册,代码如下:

    EventBus.getDefault().unregister(this);

发送消息(发布者Publisher),post方法后面的参数可以传递基本数据类型或者对象,建议传递一个对象。

         EventBus.getDefault().post(new MessageEvent(100));

接收消息(发布者Subscriber),onMessageEvent是接收消息的方法,这个方法必须添加注解@Subscribe,不然发送消息时找不到内存中注解对应的onMessageEvent方法,其次onMessageEvent方法必须有且只有一个形式参数,这个参数表示要接收的消息类型,如果发送的消息对象非MessageEvent对象,那么onMessageEvent方法是无法被执行的。

/**
 * 接收消息
 * @param event
 */
@Subscribe(threadMode = ThreadMode.MAIN)
public void onMessageEvent(MessageEvent event) {
    /* Do something */
    Log.d("yunchong", "MessageEvent的flag值为:"+event.getFlag());
}

[线程模式]

接收消息的方法被@Subscribe修饰,它是EventBus框架的自定义注解,源码如下:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Subscribe {
    ThreadMode threadMode() default ThreadMode.POSTING;

    /**
     * If true, delivers the most recent sticky event (posted with
     * {@link EventBus#postSticky(Object)}) to this subscriber (if event available).
     */
    boolean sticky() default false;

    /** Subscriber priority to influence the order of event delivery.
     * Within the same delivery thread ({@link ThreadMode}), higher priority subscribers will receive events before
     * others with a lower priority. The default priority is 0. Note: the priority does *NOT* affect the order of
     * delivery among subscribers with different {@link ThreadMode}s! */
    int priority() default 0;
}

从源码中得到的信息可以看出,该注解只能作用于方法上,并且该方法在运行时被常驻到内存中。
它有三个成员:threadMode、sticky、priority

priority:消息接收的优先级,默认为0,优先级越高就越快收到消息;
sticky:粘性事件的开关,默认值是false
threadMode:线程模式,默认模式是ThreadMode.POSTING,线程模式分为:POSTINGMAINMAIN_ORDEREDBACKGROUNDASYNC

  • POSTING:
    默认的模式,开销最小的模式,因为声明为POSTING的订阅者会在发布的同一个线程调用,发布者在主线程那么订阅者也就在主线程,反之亦,避免了线程切换,如果不确定是否有耗时操作,谨慎使用,因为可能是在主线程发布。

  • MAIN
    主线程调用,视发布线程不同处理不同,如果发布者在主线程那么直接调用(非阻塞式),如果发布者不在主线程那么阻塞式调用。

  • MAIN_ORDERED
    和MAIN差不多,主线程调用,和MAIN不同的是他保证了post是非阻塞式的(默认走MAIN的非主线程的逻辑,所以可以做到非阻塞)

  • BACKGROUND
    在子线程调用,如果发布在子线程那么直接在发布线程调用,如果发布在主线程那么将开启一个子线程来调用,这个子线程是阻塞式的,按顺序交付所有事件,所以也不适合做耗时任务,因为多个事件共用这一个后台线程。

  • ASYNC
    在子线程调用,总是开启一个新的线程来调用,适用于做耗时任务,比如数据库操作,网络请求等,不适合做计算任务,会导致开启大量线程。

[粘性事件]

EventBus的一般用法是:先注册,再定义接收事件的方法,之后发送的事件才能接收到。
但是,在Android中有写场景不符合EventBus的一般用法,比如:Fragment之间切换时,将数据从一个Fragment传递到另一个Fragment。还有一个场景是,从一个Activity跳转到另一个Activity,将数据从一个Activity传递到另一个Activity。

为了解决这个问题,EventBus推出了粘性事件。假如ActivityA跳转到ActivityB时,将数据从A发送到B,那么该如何实现呢?

第一步,在ActivityA中发送粘性事件

            //发送消息
            EventBus.getDefault().postSticky(new MessageEvent(100));

postSticky方法发送一个粘性事件。

第二步,在ActivityB中注册和反注册

    //注册
    EventBus.getDefault().register(this);

    //反注册
    EventBus.getDefault().unregister(this);

第三步,在ActivityB中定义接收事件的方法

/**
 * 接收消息
 * @param event
 */
@Subscribe(threadMode = ThreadMode.MAIN, sticky = true)
public void onMessageEvent(MessageEvent event) {
    /* Do something */
    Log.d("yunchong", "MessageEvent的flag值为:"+event.getFlag());
}

其中,sticky属性必须设置为true。

[代码混淆]

根据EventBus作者的意思,如果项目开启混淆功能的话,还需要添加以下混淆代码:

-keepattributes *Annotation*
-keepclassmembers class * {
    @org.greenrobot.eventbus.Subscribe <methods>;
}
-keep enum org.greenrobot.eventbus.ThreadMode { *; }

# And if you use AsyncExecutor:
-keepclassmembers class * extends org.greenrobot.eventbus.util.ThrowableFailureEvent {
    <init>(java.lang.Throwable);
}

[本章完...]

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