美文网首页app连接网络那些事Android应用开发那些事
移动架构<第六篇>:EventBus事件总线框架

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

作者: NoBugException | 来源:发表于2020-05-30 17:52 被阅读0次

    在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);
    }
    

    [本章完...]

    相关文章

      网友评论

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

        本文链接:https://www.haomeiwen.com/subject/tnvtzhtx.html