位置: 编程技术 - 正文

android开发之EventBus源码解析(android开发指南)

编辑:rootadmin
1. 功能介绍1.1 EventBus

推荐整理分享android开发之EventBus源码解析(android开发指南),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:android开发指南,android开发之apritag,android开发详解,android开发最全教程,android开发详解,android开发fragment,android开发范例实战宝典,android event,内容如对您有帮助,希望把文章链接给更多的朋友!

EventBus 是一个 Android 事件发布/订阅框架,通过解耦发布者和订阅者简化 Android 事件传递,这里的事件可以理解为消息,本文中统一称为事件。事件传递既可用于 Android 四大组件间通讯,也可以用户异步线程和主线程间通讯等等。

传统的事件传递方式包括:Handler、BroadCastReceiver、Interface 回调,相比之下 EventBus 的优点是代码简洁,使用简单,并将事件发布和订阅充分解耦。

1.2 概念

事件(Event):又可称为消息,本文中统一用事件表示。其实就是一个对象,可以是网络请求返回的字符串,也可以是某个开关状态等等。事件类型(EventType)指事件所属的 Class。

事件分为一般事件和 Sticky 事件,相对于一般事件,Sticky 事件不同之处在于,当事件发布后,再有订阅者开始订阅该类型事件,依然能收到该类型事件最近一个 Sticky 事件。

订阅者(Subscriber):订阅某种事件类型的对象。当有发布者发布这类事件后,EventBus 会执行订阅者的 onEvent 函数,这个函数叫事件响应函数。订阅者通过 register 接口订阅某个事件类型,unregister 接口退订。订阅者存在优先级,优先级高的订阅者可以取消事件继续向优先级低的订阅者分发,默认所有订阅者优先级都为 0。

发布者(Publisher):发布某事件的对象,通过 post 接口发布事件。

2. 总体设计

本项目较为简单,总体设计请参考3.1 订阅者、发布者、EventBus 关系图 及 4.1 类关系图。

3. 流程图3.1 订阅者、发布者、EventBus 关系图

EventBus 负责存储订阅者、事件相关信息,订阅者和发布者都只和 EventBus 关联。

3.2 事件响应流程

订阅者首先调用 EventBus 的 register 接口订阅某种类型的事件,当发布者通过 post 接口发布该类型的事件时,EventBus 执行调用者的事件响应函数(onEvent函数)。

4. 详细设计4.1 类关系图

以上是 EventBus 主要类的关系图,从中我们也可以看出大部分类都与 EventBus 直接关联。上部分主要是订阅者相关信息,中间是 EventBus 类,下面是 发布者发布事件后的调用,具体类的功能请看下面的详细介绍。

4.2 类详细介绍4.2.1 EventBus.java

EventBus 类负责所有对外暴露的 API,其中的 register、post、unregister 函数配合上自定义的 EventType 及事件响应函数即可完成核心功能,见 3.2 图。

EventBus 默认可通过静态函数 getDefault 获取单例,当然有需要也可以通过 EventBusBuilder 或 构造函数新建一个 EventBus,每个新建的 EventBus 发布和订阅事件都是相互隔离的,即一个 EventBus 对象中的发布者发布事件,另一个 EventBus 对象中的订阅者不会收到该订阅。

EventBus 中对外 API,主要包括三类:register() 和 unregister()、post()、onEvent()

(1) register 和 unregister 分别表示订阅事件和取消订阅。register 最底层函数有三个参数,分别为订阅者对象、是否是 Sticky 事件、优先级。

PS:在此之前的版本 EventBus 还允许自定义事件响应函数名称,这版本中此功能已经被去除。

register 函数流程图如下:

register 函数中会先根据订阅者类名去subscriberMethodFinder中查找当前订阅者所有事件响应函数,然后循环每一个事件响应函数,依次执行下面的 subscribe 函数:

(2) subscribe

subscribe 函数分三步:

第一步:通过subscriptionsByEventType得到该事件类型所有订阅者信息队列,根据优先级将当前订阅者信息插入到订阅者队列subscriptionsByEventType中;

第二步:在typesBySubscriber中得到当前订阅者订阅的所有事件队列,将此事件保存到队列typesBySubscriber中,用于后续取消订阅;

第三步:检查这个事件是否是 Sticky 事件,如果是则从stickyEvents事件保存队列中取出该事件类型最后一个事件发送给当前订阅者。

(3) post、cancel 、removeStickyEvent

post 函数用于发布事件;cancel 函数用于取消某订阅者订阅的所有事件类型;removeStickyEvent 函数用于删除 sticky 事件。

post 函数流程图如下:

post函数流程解析:

post 函数会首先得到当前线程的 post 信息PostingThreadState,其中包含事件队列,将当前事件添加到其事件队列中,然后循环调用postSingleEvent 函数发布队列中的每个事件。

postSingleEvent 函数会先去eventTypesCache得到该事件对应类型的的父类及接口类型,没有缓存则查找并插入缓存。循环得到的每个类型和接口,调用postSingleEventForEventType 函数发布每个事件到每个订阅者。

postSingleEventForEventType 函数在subscriptionsByEventType查找该事件订阅者订阅者队列,调用 postToSubscription 函数向每个订阅者发布事件。

postToSubscription 函数中会判断订阅者的 ThreadMode,从而决定在什么 Mode 下执行事件响应函数。

ThreadMode共有四类:

PostThread:默认的 ThreadMode,表示在执行 Post 操作的线程直接调用订阅者的事件响应方法,不论该线程是否为主线程(UI线程)。当该线程为主线程时,响应方法中不能有耗时操作,否则有卡主线程的风险。适用场景:对于是否在主线程执行无要求,但若 Post 线程为主线程,不能耗时的操作;

MainThread:在主线程中执行响应方法。如果发布线程就是主线程,则直接调用订阅者的事件响应方法,否则通过主线程的 Handler 发送消息在主线程中处理——调用订阅者的事件响应函数。显然,MainThread类的方法也不能有耗时操作,以避免卡主线程。适用场景:必须在主线程执行的操作;

BackgroundThread:在后台线程中执行响应方法。如果发布线程不是主线程,则直接调用订阅者的事件响应函数,否则启动唯一的后台线程去处理。由于后台线程是唯一的,当事件超过一个的时候,它们会被放在队列中依次执行,因此该类响应方法虽然没有PostThread类和MainThread类方法对性能敏感,但最好不要有重度耗时的操作或太频繁的轻度耗时操作,以造成其他操作等待。适用场景:操作轻微耗时且不会过于频繁,即一般的耗时操作都可以放在这里;

Async:不论发布线程是否为主线程,都使用一个空闲线程来处理。和BackgroundThread不同的是,Async类的所有线程是相互独立的,因此不会出现卡线程的问题。适用场景:长耗时操作,例如网络访问。

(4) 主要成员变量含义:

1.defaultInstance默认的 EventBus 实例,根据EventBus.getDefault()函数得到。

2.DEFAULT_BUILDER默认的 EventBus Builder。

3.eventTypesCache事件对应类型及其父类和实现的接口的缓存,以 eventType 为 key,元素为 Object 的 ArrayList 为 Value,Object 对象为 eventType 的父类或接口。

android开发之EventBus源码解析(android开发指南)

4.subscriptionsByEventType事件订阅者的保存队列,以 eventType 为 key,元素为Subscription的 ArrayList 为 Value,其中Subscription为订阅者信息,由 subscriber, subscriberMethod, priority 构成。

5.typesBySubscriber订阅者订阅的事件的保存队列,以 subscriber 为 key,元素为 eventType 的 ArrayList 为 Value。

6.stickyEventsSticky 事件保存队列,以 eventType 为 key,event 为元素,由此可以看出对于同一个 eventType 最多只会有一个 event 存在。

7.currentPostingThreadState当前线程的 post 信息,包括事件队列、是否正在分发中、是否在主线程、订阅者信息、事件实例、是否取消。

8.mainThreadPoster、backgroundPoster、asyncPoster事件主线程处理者、事件 Background 处理者、事件异步处理者。

9.subscriberMethodFinder订阅者响应函数信息存储和查找类。

.executorService异步和 BackGround 处理方式的线程池。

.throwSubscriberException当调用事件处理函数异常时是否抛出异常,默认为 false,建议通过

打开。

.logSubscriberExceptions当调用事件处理函数异常时是否打印异常信息,默认为 true。

.logNoSubscriberMessages当没有订阅者订阅该事件时是否打印日志,默认为 true。

.sendSubscriberExceptionEvent当调用事件处理函数异常时是否发送 SubscriberExceptionEvent 事件,若此开关打开,订阅者可通过

订阅该事件进行处理,默认为 true。

.sendNoSubscriberEvent当没有事件处理函数对事件处理时是否发送 NoSubscriberEvent 事件,若此开关打开,订阅者可通过

订阅该事件进行处理,默认为 true。

.eventInheritance是否支持事件继承,默认为 true。

4.2.2 EventBusBuilder.java

跟一般 Builder 类似,用于在需要设置参数过多时构造 EventBus。包含的属性也是 EventBus 的一些设置参数,意义见4.2.1 EventBus.java的介绍,build 函数用于新建 EventBus 对象,installDefaultEventBus 函数将当前设置应用于 Default EventBus。

4.2.3 SubscriberMethodFinder.java

订阅者响应函数信息存储和查找类,由 HashMap 缓存,以 ${subscriberClassName} 为 key,SubscriberMethod 对象为元素的 ArrayList 为 value。findSubscriberMethods 函数用于查找订阅者响应函数,如果不在缓存中,则遍历自己的每个函数并递归父类查找,查找成功后保存到缓存中。遍历及查找规则为:

a. 遍历 subscriberClass 每个方法;

b. 该方法不以java.、javax.、android.这些 SDK 函数开头,并以onEvent开头,表示可能是事件响应函数继续,否则检查下一个方法;

c. 该方法是否是 public 的,并且不是 ABSTRACT、STATIC、BRIDGE、SYNTHETIC 修饰的,满足条件则继续。其中 BRIDGE、SYNTHETIC 为编译器生成的一些函数修饰符;

d. 该方法是否只有 1 个参数,满足条件则继续;

e. 该方法名为 onEvent 则 threadMode 为ThreadMode.PostThread;

该方法名为 onEventMainThread 则 threadMode 为ThreadMode.MainThread; 该方法名为 onEventBackgroundThread 则 threadMode 为ThreadMode.BackgroundThread; 该方法名为 onEventAsync 则 threadMode 为ThreadMode.Async;

其他情况且不在忽略名单 (skipMethodVerificationForClasses) 中则抛出异常。

f. 得到该方法唯一的参数即事件类型 eventType,将这个方法、threadMode、eventType 一起构造 SubscriberMethod 对象放到 ArrayList 中。

g. 回到 b 遍历 subscriberClass 的下一个方法,若方法遍历结束到 h; h. 回到 a 遍历自己的父类,若父类遍历结束回到 i;

i. 若 ArrayList 依然为空则抛出异常,否则会将 ArrayList 做为 value,${subscriberClassName} 做为 key 放到缓存 HashMap 中。 对于事件函数的查找有两个小的性能优化点:

a. 第一次查找后保存到了缓存中,即上面介绍的 HashMap

b. 遇到 java. javax. android. 开头的类会自动停止查找

类中的 skipMethodVerificationForClasses 属性表示跳过哪些类中非法以 onEvent 开头的函数检查,若不跳过则会抛出异常。

PS:在此之前的版本 EventBus 允许自定义事件响应函数名称,缓存的 HashMap key 为 ${subscriberClassName}.${eventMethodName},这版本中此功能已经被去除。

4.2.4 SubscriberMethod.java

订阅者事件响应函数信息,包括响应方法、线程 Mode、事件类型以及一个用来比较 SubscriberMethod 是否相等的特征值 methodString 共四个变量,其中 methodString 为${methodClassName}#${methodName}(${eventTypeClassName}。

4.2.5 Subscription.java

订阅者信息,包括 subscriber 对象、事件响应方法 SubscriberMethod、优先级 priority。

4.2.6 HandlerPoster.jva

事件主线程处理,对应ThreadMode.MainThread。继承自 Handler,enqueue 函数将事件放到队列中,并利用 handler 发送 message,handleMessage 函数从队列中取事件,invoke 事件响应函数处理。

4.2.7 AsyncPoster.java

事件异步线程处理,对应ThreadMode.Async,继承自 Runnable。enqueue 函数将事件放到队列中,并调用线程池执行当前任务,在 run 函数从队列中取事件,invoke 事件响应函数处理。

4.2.8 BackgroundPoster.java

事件 Background 处理,对应ThreadMode.BackgroundThread,继承自 Runnable。enqueue 函数将事件放到队列中,并调用线程池执行当前任务,在 run 函数从队列中取事件,invoke 事件响应函数处理。与 AsyncPoster.java 不同的是,BackgroundPoster中的任务只在同一个线程中依次执行,而不是并发执行。

4.2.9 PendingPost.java

订阅者和事件信息实体类,并含有同一队列中指向下一个对象的指针。通过缓存存储不用的对象,减少下次创建的性能消耗。

4.2. PendingPostQueue.java

通过 head 和 tail 指针维护一个PendingPost队列。HandlerPoster、AsyncPoster、BackgroundPoster 都包含一个此队列实例,表示各自的订阅者及事件信息队列,在事件到来时进入队列,处理时从队列中取出一个元素进行处理。

4.2. SubscriberExceptionEvent.java

当调用事件处理函数异常时发送的 EventBus 内部自定义事件,通过 post 发送,订阅者可自行订阅这类事件进行处理。

4.2. NoSubscriberEvent.java

当没有事件处理函数对事件处理时发送的 EventBus 内部自定义事件,通过 post 发送,订阅者可自行订阅这类事件进行处理。

4.2. EventBusException.java

封装于 RuntimeException 之上的 Exception,只是覆盖构造函数,相当于一个标记,标记是属于 EventBus 的 Exception。

4.2. ThreadMode.java

线程 Mode 枚举类,表示事件响应函数执行线程信息,包括ThreadMode.PostThread、ThreadMode.MainThread、ThreadMode.BackgroundThread、ThreadMode.Async四种。

Demo地址:EventBus Demo

Android平台上MVP的介绍 原文链接:IntroductiontoModel-View-PresenteronAndroid原文作者:konmik译文出自:其他

androidday3 ##_android下数据库的创建(重点)在Android平台上,集成了一个嵌入式关系型数据库—SQLite,SQLite3支持NULL、INTEGER、REAL(浮点数字)、TEXT(字符串文本)和

[置顶] Android GradientDrawable高级应用 以后完全用不上美工了 先看截图1图为自定义的Textview2、3图为点击效果具体实现如下:1.定义自定义控件属性declare-styleablename=ShapeTextviewattrname=touchSolidColorformat=color/attrname=solidCo

标签: android开发指南

本文链接地址:https://www.jiuchutong.com/biancheng/386098.html 转载请保留说明!

上一篇:Android源码下载与编译:(android源码下载地址)

下一篇:Android平台上MVP的介绍(android mvp mvc)

  • 如何理解递延所得税资产和递延所得税负债
  • 月收入10万以下免税哪些企业适用
  • 交通费中的高速费怎么算
  • 建筑工程机械服务有限公司企业
  • 公司网站建设费做账是流量应计入
  • 企业所得税季度申报时间
  • 小规模升一般纳税人怎么操作
  • 安装服务费发票需要备注地址吗
  • 工商公示纳税总额包含个税吗
  • 房屋租赁收入怎么做账
  • 进口转卖业务
  • 企业购房税费政策
  • 开票方没缴税咋办
  • 电子商务支付平台有哪些
  • 保证金可以抵货款吗
  • 开发票时怎么修改税收分类简称?
  • 委托代征申报
  • 国家金库为什么扣我银行里的钱
  • 建筑企业成本票是含税价还是不含税价
  • 我们4月10日
  • 期间费用包括哪些?
  • 海关对现金携带数量有要求吗
  • 购设备发票只开来一部分怎么入账?
  • 应交税费有余额怎么结转
  • 已确定收入后补缴增值税
  • 注册资本与利息的税前扣除
  • 管理费是否需合计
  • 人力成本费用率和人工成本利润率
  • 什么是电流定律
  • 苹果a1586是什么配置
  • 以太网没有internet
  • 发放应付职工薪酬的账务处理
  • php require include
  • 主营业务收入会计分录例题
  • html文档怎么写
  • 转租会计如何记账
  • 汽车理赔款
  • 资产减值损失影响当期利润吗
  • vue开发教程
  • egi脑电数据处理
  • split 文件分割
  • 申报表中的免税销售额是收入吗
  • 没有综合所得的经营所得如何预扣预缴个税
  • 包装物为什么是无菌的
  • 帝国cms如何搭建的
  • markdown小小白常用语法
  • mongodb简单使用
  • 融资租赁的固定资产
  • 普通动产和特殊动产登记的区别
  • 不得从销项中抵扣的进项有哪些
  • 汽车维修行业工时标准
  • 计提坏账准备的方法
  • 抵消递延所得税资产会计处理
  • 机票电子行程单查询
  • 主营业务成本里的工资
  • 稽查跨年进项税额转出要调整所得税吗
  • 公司购买银行理财产品收益交什么税
  • 缴纳税款滞纳金应计入哪个科目
  • 个人打钱到公司账上违法吗
  • 其他综合收益要转入投资收益吗
  • 工会筹备金该按什么计算
  • 赠送客户样品记什么费用
  • 资本金与注册资金的关系
  • ubuntu如何安装
  • ubuntu 网络设置命令
  • centos6.5关闭休眠
  • backupnotify.exe是什么文件的进程 backupnotify进程安全吗
  • fp3是什么文件格式
  • lnmp一键安装包是什么
  • Win7系统打开D盘文件后怎么没有后退箭头
  • 不得不佩服自己的搞笑说说
  • js 箭头表达式
  • jquery 图片放大预览
  • dom操作的方法
  • javascript的代码写在哪里
  • Android SimpleAdapter
  • 保险专票可以抵进项税吗
  • 租房交税是房东交还是中介交
  • 工业生产和农业生产区别
  • 长期挂账的其他应付款的规定
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设