200行代码手写一套EventBus事件总线


理论千万篇,不如实战来一篇。

源码 https://github.com/harvie1208/EventBus

关键词:观察者模式、反射、自定义注解、线程调度

手写200行代码,一步一步实现EventBus核心功能,看完可以写一套属于自己的事件总线库啦!

不知大家平常在看博客的时候有没有和我遇到一样的问题,就是看的是懂非懂,好像懂了,又好像没懂。

主要有以下两点:

    1.文章缺少部分实现思路,导致自己实现时卡住。

    2.术语太过专业化,不易理解。

在求知的路上,我也看了不少文章,有非常优秀的,也有缺这少那的。一路走来填了不少坑,后面我会将所学知识点整理出来分享给大家,尽量做到通俗易懂的理论加完整案例源码。一方面是对自己知识点的总结回顾,另一方面也希望能帮助到有需要的同学少走弯路。因技术水平有限,如有不正之处,还望各位不吝指教。

EventBus简介

EventBus顾名思义就是事件总线,实际上就是一个`事件发布者/事件监听者(订阅者)` 的框架, 发布者发布事件,Bus自动处理与分发,监听者被动的接受。简化各种异步和跳转的通信。

![](https://user-gold-cdn.xitu.io/2019/7/18/16c059b4885766f6?w=1000&h=374&f=png&s=111792)

使用场景示例

    1.短信验证码登陆场景

        主登陆界面A->输入手机号界面B->短信验证码界面C->登陆成功跳转首页D

        需求:登陆成功后需要关闭A、B、C三个页面

    2.音乐播放场景

        假如首页有5个tab(包含5个fragment),每个fragment中都有音乐播放状态小图标

        需求:音乐播放或暂停时,需要更新所有播放状态图标

核心思路

    使用观察者模式,在需要接收事件的方法上添加订阅注解标识,并将此方法所在对象添加到订阅者集合,发送事件时遍历订阅者集合,在通过反射调用相关订阅方法。

代码实战

1.编写EventBus核心类,使用单例模式提供唯一实例

public class EventBus {

    private static EventBus myBus;

    public static EventBus getInstance(){

        if (myBus==null){

            synchronized (EventBus.class){

                if (myBus==null){

                    myBus = new EventBus();

                }

            }

        }

        return myBus;

    }

}

2.给订阅方法添加@Subscribe标识

    创建自定义注解@Subscribe用来标示订阅方法

    注解Annontation是Java5开始引入的新特征,通俗来说就是为程序的元素(类、方法、成员变量)添加标记用的

    @Target(ElementType.METHOD) //表示此注解作用域在方法上

    @Retention(RetentionPolicy.RUNTIME) //编译程序处理完注解信息后存储在class中,可由VM读入

    public @interface Subscribe {


        ThreadModel thread();//用于指定被注解方法执行时所在的线程

    }

    使用注解

    public class MainActivity extends AppCompatActivity {

        @Subscribe(thread = ThreadModel.BACKGROUND)//指定在子线程中执行

        public void haha(LoginEvent loginEvent){

            Log.e("EventBus",loginEvent.getLoginStatus()+Thread.currentThread().getName());

        }

    }

3.注册订阅关系

    * 先声明一个集合用于存储类对象和被注解的方法及线程模式

    * 遍历注册对象的所有方法,取出带有@Subscribe注解的方法

    * 获取参数类型数组,当前仅支持一个参数

    * 获取指定线程模式

    * 构建订阅者实例(方法、参数类型、线程模式),加入订阅集合

    public class EventBus {

        //存储订阅类及方法参数

        private Map<Object,List<Subscriber>> subscribeMethod;

        public void register(Object obj){

            if (obj==null){

                return;

            }

            Class<?> mclazz = obj.getClass();

            //获取本类所有方法

            Method[] methods = mclazz.getDeclaredMethods();

            List<Subscriber> methods1 = new ArrayList<>();

            for (Method method : methods){

                //获取带有我们Subscribe注解的方法

                Subscribe subscribe = method.getAnnotation(Subscribe.class);

                if (subscribe==null){

                    continue;

                }

                //获取参数类型集合

                Class<?>[] typeVariable = method.getParameterTypes();

                if (typeVariable.length!=1){

                    continue;

                }

                ThreadModel threadModel = subscribe.thread();

                Subscriber busMethod = new Subscriber(method,threadModel,typeVariable[0]);

                methods1.add(busMethod);

            }

            if (methods1.size()>0){

                subscribeMethod.put(obj,methods1);

            }

        }

    }

4.注销订阅

    将此订阅对象移除订阅集合

    public void unRegister(Object object){

        if (subscribeMethod.containsKey(object)){

            subscribeMethod.remove(object);

        }

    }

5.发送事件

    * 根据发送事件参数类型,遍历集合找到对应方法

    * 判断线程模式,主线程用handler处理,子线程用线程池处理

    * 反射调用方法将事件传过去

    public class EventBus {

        //存储订阅类及方法参数

        private Map<Object,List<Subscriber>> subscribeMethod;

        //线程调度

        private Handler mHandler;

        //线程池

        private ExecutorService executorService;

        private EventBus(){

            subscribeMethod = new HashMap<>();

            mHandler = new Handler(Looper.getMainLooper());

            executorService = Executors.newCachedThreadPool();

        }

        public void postEvent(Object eventParam){

            Set<Object> set = subscribeMethod.keySet();

            Iterator<Object> iterable =set.iterator();

            while (iterable.hasNext()){

                Object obj = iterable.next();

                List<Subscriber> busMethodList = subscribeMethod.get(obj);

                for (Subscriber busMethod : busMethodList){

                    if(busMethod.getParamsType() == eventParam.getClass()){

                        invoke(obj,busMethod,eventParam);

                    }

                }

            }

        }

        private void invoke(final Object obj, final Subscriber busMethod, final Object eventParam){

            switch (busMethod.getThreadModel()){

                case MAIN:

                    //通过handler调度到主线程

                    mHandler.post(new EventRunable(busMethod, obj, eventParam));

                    break;

                default:

                    //交由线程池处理

                    executorService.execute(new EventRunable(busMethod, obj, eventParam));

                    break;

            }

        }

    }

    事件参数与接收参数类型一致即可,方法名随意

    EventBus.getInstance().postEvent(new LoginEvent("登录成功"));

## 总结

很多看似高大上的框架其实也没我们想的那么难,写着写着就顺手了,知而不行为不知,快动起手来吧!

源码 https://github.com/harvie1208/EventBus

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

推荐阅读更多精彩内容