代码里面发送粘性事件代码如下:

  1. //  发送Sticky事件
    EventBus.getDefault().postSticky(new User("soyoungboy", "西安财经学院"),
    "soyoungboy");


然后我们进入postSticky方法里面去:
EventType 是什么?
该类是描述一个函数唯一性的对象,参数类型、tag两个条件保证了对象的唯一性.通过该类的对象来查找注册了相应类型和tag的所有订阅者{@see* Subscription}, 并且在接到消息时调用所有订阅者对应的函数. 
mStickyEvents是什么?
  1.  private List<EventType> mStickyEvents = Collections
    .synchronizedList(new LinkedList<EventType>());


是个list集合,用于存储eventType的list集合。
  1.  /**
    * 发布Sticky事件,tag为EventType.DEFAULT_TAG
    *
    * @param event
    */
    public void postSticky(Object event) {
    postSticky(event, EventType.DEFAULT_TAG);
    }
    /**
    * 发布含有tag的Sticky事件
    *
    * @param event 事件
    * @param tag 事件tag
    */
    public void postSticky(Object event, String tag) {
    EventType eventType = new EventType(event.getClass(), tag);
    eventType.event = event;
    mStickyEvents.add(eventType);
    // 处理sticky事件
    // mDispatcher.handleStickyEvent(eventType, null);
    }
这是发布,如何接收粘性事件,我们也看下源码来分析下:
首先要注册粘性事件的eventbus:
  1. EventBus.getDefault().registerSticky(this);
走进registerSticky代码里面看下:
  1.  /**
    * 以sticky的形式注册,则会在注册成功之后迭代所有的sticky事件
    *
    * @param subscriber
    */
    public void registerSticky(Object subscriber) {
    this.register(subscriber);
    // 处理sticky事件
    mDispatcher.dispatchStickyEvents(subscriber);
    }

    register代码不进行分析,因为前面讲过,就是将@ subscriber的订阅者对象放入map集合里面去

重点分析下 mDispatcher.dispatchStickyEvents(subscriber);
  1.  void dispatchStickyEvents(Object subscriber) {
    for (EventType eventType : mStickyEvents) {
    handleStickyEvent(eventType, subscriber);
    }
    }


在往下面走:
根据不同的ThreadMode去调用不同的handler(这里是事件分发器对象)的handleEvent方法。
ThreadMode是在代码里面通过注解传进来的。
/**
* 处理单个Sticky事件
*
* @param eventType
* @param aEvent
*/
private void handleStickyEvent(EventType eventType, Object subscriber) {
//从缓存获取eventTypes内容
List<EventType> eventTypes = getMatchedEventTypes(eventType, eventType.event);
// 事件
Object event = eventType.event;
//循环遍历eventType集合
for (EventType foundEventType : eventTypes) {
Log.e("", "### 找到的类型 : " + foundEventType.paramClass.getSimpleName()
+ ", event class : " + event.getClass().getSimpleName());
//从订阅者的map集合里面获取指定的订阅者集合的list集合,根据eventType
final List<Subscription> subscriptions = mSubcriberMap.get(foundEventType);
if (subscriptions == null) {
continue;
}
//遍历list集合
for (Subscription subItem : subscriptions) {
final ThreadMode mode = subItem.threadMode;
EventHandler eventHandler = getEventHandler(mode);
// 如果订阅者为空,那么该sticky事件分发给所有订阅者.否则只分发给该订阅者
if (isTarget(subItem, subscriber)
&& (subItem.eventType.equals(foundEventType)
|| subItem.eventType.paramClass
.isAssignableFrom(foundEventType.paramClass))) {
// 处理事件
eventHandler.handleEvent(subItem, event);
}
}
}
}
mDispatcher是个事件分发器;
EventDispatcher事件分发器中有三条个ThreadEventHandler:
UIThreadEventHandler
DefaultEventHandler
AsyncEventHandler
都实现了EventHandler事件处理接口,实现handleEvent方法:
UIThreadEventHandler中实现:
事件处理在UI线程,通过Handler将事件处理post到UI线程的消息队列,UIThreadEventHandler的handleEvent指向DefaultEventHandler的handleEvent方法;列  

  1. /**
    * @param subscription
    * @param event
    */
    public void handleEvent(final Subscription subscription, final Object event) {
    mUIHandler.post(new Runnable() {
    @Override
    public void run() {
    mEventHandler.handleEvent(subscription, event);
    }
    });
    }


 
AsyncEventHandler中实现:默认也是子类为DefaultEventHandler的handleEvent去实现,重点还是DefaultEventHandler中的handleEvent的实现。
  1.  /**
    * 将订阅的函数执行在异步线程中
    *
    * @param subscription
    * @param event
    */
    public void handleEvent(final Subscription subscription, final Object event) {
    mDispatcherThread.post(new Runnable() {
    @Override
    public void run() {
    mEventHandler.handleEvent(subscription, event);
    }
    });
    }


DefaultEventHandler中实现:
重点是subscription.targetMethod.invoke(subscription.subscriber.get(), event),代码走到最后走到这里,那么到这这里执行了什么操作呢;
执行反射操作,根据参数查找方法,进行反射调用。
 /**
* handle the event
*
* @param subscription
* @param event
*/
public void handleEvent(Subscription subscription, Object event) {
if (subscription == null
|| subscription.subscriber.get() == null) {
return;
}
try {
// 执行
subscription.targetMethod.invoke(subscription.subscriber.get(), event);
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}

【第五篇】androidEventbus源代码阅读和分析之发送粘性事件和接收粘性事件代码分析的更多相关文章

  1. 【第五篇】androidEventbus源代码阅读和分析之unregister代码分析

    代码里面注销eventbus一般我们会在onDestory里面这么写: EventBus.getDefault().unregister(this); 然后走到unregister里面去看看: /** ...

  2. 【第四篇】androidEventbus源代码阅读和分析

    1,分析androidEventbus的注册源代码: 我们在使用androidEventbus的第一步是注册eventbus,如下代码: EventBus.getDefault().register( ...

  3. Android4.0图库Gallery2代码分析(二) 数据管理和数据加载

    Android4.0图库Gallery2代码分析(二) 数据管理和数据加载 2012-09-07 11:19 8152人阅读 评论(12) 收藏 举报 代码分析android相册优化工作 Androi ...

  4. tensorflow笔记:多层LSTM代码分析

    tensorflow笔记:多层LSTM代码分析 标签(空格分隔): tensorflow笔记 tensorflow笔记系列: (一) tensorflow笔记:流程,概念和简单代码注释 (二) ten ...

  5. PostgreSQL代码分析,查询优化部分,pull_ands()和pull_ors()

    PostgreSQL代码分析,查询优化部分. 这里把规范谓词表达式的部分就整理完了,阅读的顺序例如以下: 一.PostgreSQL代码分析,查询优化部分,canonicalize_qual 二.Pos ...

  6. 免费的Lucene 原理与代码分析完整版下载

    Lucene是一个基于Java的高效的全文检索库.那么什么是全文检索,为什么需要全文检索?目前人们生活中出现的数据总的来说分为两类:结构化数据和非结构化数据.很容易理解,结构化数据是有固定格式和结构的 ...

  7. PostgreSQL代码分析,查询优化部分,canonicalize_qual

    这里把规范谓词表达式的部分就整理完了.阅读的顺序例如以下: 一.PostgreSQL代码分析,查询优化部分,canonicalize_qual 二.PostgreSQL代码分析,查询优化部分,pull ...

  8. tensorflow笔记:多层CNN代码分析

    tensorflow笔记系列: (一) tensorflow笔记:流程,概念和简单代码注释 (二) tensorflow笔记:多层CNN代码分析 (三) tensorflow笔记:多层LSTM代码分析 ...

  9. Linux kernel的中断子系统之(七):GIC代码分析

    返回目录:<ARM-Linux中断系统>. 总结: 原文地址:<linux kernel的中断子系统之(七):GIC代码分析> 参考代码:http://elixir.free- ...

随机推荐

  1. Python中的多进程与多线程(二)

    在上一章中,学习了Python多进程编程的一些基本方法:使用跨平台多进程模块multiprocessing提供的Process.Pool.Queue.Lock.Pipe等类,实现子进程创建.进程池(批 ...

  2. 日志快速筛选 之 linux命令grep|uniq|wc|awk

    以前我个人的观念是,在线上运行的东西尽量不要记什么流水日志. 但是后来我变了,发现在线上记日志是一个绝对有必要的东西,尤其是在当下很流行的微服务的推动下,没有日志的帮助,犹如一个睁眼瞎,排查问题基本靠 ...

  3. Flexible 弹性盒子模型之CSS justify-content 属性

    实例 在弹性盒对象的 <div> 元素中的各项周围留有空白: div { display: flex; justify-content: space-around; } 复制 效果预览 浏 ...

  4. SAP CRM 为用户创建业务合作伙伴并分配到组织单位

    想要在SAP CRM的前台完成一些操作,需要登录的用户在系统中存在对应的业务合作伙伴才可以,某些情况下,还需要被分配到正确的公司.部门.职位.下面是相关的操作步骤. 本文假定读者已经拥有一个开发帐号. ...

  5. UITabelview的删除

    删除的效果 Automatic Bottom Fade left middle none right top 简单删除 先删除数据源里的数据,然后再删除cell,否者会报错 let indexPath ...

  6. DB2表是否存在

    select count(1) from syscat.tables where tabname='T1';

  7. yarn的调度器

    三种调度器 1.FIFO Scheduler 把应用按提交的顺序排成一个队列,这是一个先进先出队列,在进行资源分配的时候,先给队列中最头上的应用进行分配资源,等最前面的应用需求满足后再给下一个分配,以 ...

  8. QT学习笔记—1

    1.模态和非模态的区别:非模态可以同时操作两个窗口,模态的只能在顶层窗口关闭之后才能使用其他窗口 //同时显示出widget和dialog窗口,非模态     QDialog *dialog = ne ...

  9. [Q]无法卸载怎么办

    正确卸载CAD批量打图精灵的方法是进入操作系统,“控制面版”,然后运行“添加和删除程序”,找到CAD批量打图精灵,选“更改/删除”,按照提示操作,即可进行卸载. 若使用强制卸载工具(如360等)卸载可 ...

  10. 什么是Bash Shell的内建(build in)命令

    1.什么是build in命令: shell内建命令是指bash(或其它版本)工具集中的命令.一般都会有一个与之同名的系统命令,比如bash中的echo命令与/bin/echo是两个不同的命令,尽管他 ...