当一个Android应用功能越来越多的时候。app中各个部分之间通信。往往採用Observer的方式来进行,即注冊----通知----注销的方式运行 各类控件常常须要依据某个状态来更新显示内容。这样的场景常见的解决方案就是定义一个接口,须要关注该事件的控件来实现这个接口。 接口类:
      public interface OnChangedListener {
            void onDataChanged();
        }

       被观察者往往以例如以下形式实现:

public abstract class AbsHTTPRequest {
private final WeakHashMap<OnChangedListener, Boolean> mListeners = new WeakHashMap<OnChangedListener, Boolean>();
public interface OnChangedListener {
void onDataChanged();
} /*HTTP's response*/
public abstract void onResponse(); public final void addListener(OnChangedListener listener) {
mListeners.put(listener, true);
} public final void removeListener(OnChangedListener listener) {
mListeners.remove(listener);
} protected final void notifyDataChanged() {
Set<OnChangedListener> keys = mListeners.keySet();
if(keys != null) {
Iterator<OnChangedListener> iterator = keys.iterator();
while(iterator.hasNext()) {
iterator.next().onDataChanged();
}
}
}
}

详细的主题角色( 被观察者),实现方式例如以下:

public class LoginRequest extends AbsHTTPRequest implements OnChangedListener{
public void onResponse(){
addListener(this);
notifyDataChanged();
} @Override
public void onDataChanged() {
// TODO Auto-generated method stub
System.out.println("LoginRequest");
}
}

使用观察者模式有一个弊病就是部件之间的耦合度太高,全部的主题角色都须要实现同一个interface。

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同一时候监听某一个主题对象。

假设主题角色被注冊的observer越多。那么须要实现的interface也就越多,接口方法数量也就越多。

     怎样来进行解耦,让代码逻辑更清晰。可读性更强。是一个问题。

在Android中也有一个类似功能的开源库EventBus。能够非常方便的帮助我们实现观察者模式,而且让各个组件之间的耦合性更低。

      关于EventBus的解说文章,网络业非常多,这里推荐一篇,解说比較具体的blog,http://www.cnblogs.com/angeldevil/p/3715934.html 
对EventBus的认识,不妨从demo入手。先易后难。

首先知道怎样使用,然后再深究源代码,才干循序渐进,吃透当中的设计理念。便于日后的代码调试和模块重构。关于demo,网上有非常多,能够自己去查收。

      EventBus的使用有4个步奏:
1.定义事件类型:
    public class MyEvent
2.注冊订阅者:
    EventBus.getDefault().register(this)
3.发送事件:
    EventBus.getDefault().post(new MyEvent())
4.接收事件,处理
     订阅者回调的函数。官方指导。函数名称以onEvent开头。

    在EventBus模块中,有几个重要的概念。了解了这几个概念后,也就不难懂了。

          Event:能够是随意类型的对象
          Subscriber:订阅者,接收特定的
          Publisher:公布者。用于通知Subscriber发送

EventType:onEvent函数中的參数。表示事件对象,用户自己定义的。

          Subscriber:订阅源,即调用register注冊的对象,这个对象内包括onEvent成员函数。
SubscribMethod.java
final class SubscriberMethod {
final Method method; /*Method类型的method成员表示这个onEvent,即事件处理函数。 同一时候也包括订阅源*/
final ThreadMode threadMode;
final Class<?> eventType; /*事件的对象,用户自己定义Object*/
... ... ... ... ... ... ... ... ... ... ... ...
}

Subscription.java

final class Subscription {
final Object subscriber; /*订阅源Subscriber,即调用register注冊的对象*/
final SubscriberMethod subscriberMethod; /**/
final int priority;
... ... ... ... ... ... ... ... ... ... ... ...
}
  类EventBus中。有两个核心的成员
   /*EventType -> List<Subscription>,事件到订阅对象之间的映射*/
private final Map<Class<?>, CopyOnWriteArrayList<Subscription>> subscriptionsByEventType;
/* Subscriber -> List<EventType>,订阅对象到它订阅的的全部事件的映射关系*/
private final Map<Object, List<Class<?>>> typesBySubscriber;

  注冊流程:在调用register函数时,EventBus类有多个重载的register函数,可是作者更倾向于使用register(this);含有 多个參数的register函数中,明白标注了@deprecated,原创作者不建议使用。

从代码:

    public void register(Object subscriber) {
register(subscriber, DEFAULT_METHOD_NAME, false, 0);
}

能够观察到,全部重载的register函数。都调用到了

    private synchronized void register(Object subscriber, String methodName, boolean sticky, int priority) {
List<SubscriberMethod> subscriberMethods = subscriberMethodFinder.findSubscriberMethods(subscriber.getClass(),methodName);
for (SubscriberMethod subscriberMethod : subscriberMethods) {
subscribe(subscriber, subscriberMethod, sticky, priority);
}
}

当中注冊函数register。默认參数DEFAULT_METHOD_NAME为函数名称"onEvent",在java放射机制中,全部的事件处理函数名称 统一为“onEvent”,只參数不一致。onEvent的參数为用户自己定义的对象。

       注冊时。使用SubscriberMethodFinder的对象,调用到findSubscriberMethods方法。获取到List<SubscriberMethod>。

     数组对象Method[],调用getMethods()方法, 获取的是类的全部共同拥有方法,这就包含自身的全部public方法。和从基类继承的、从接口实现的全部public方法。

这也是为啥,我们的onEvent函数,要定义为public方法的原因哦。

    在findSubscriberMethods函数中,进行如此频繁的遍历。就是为了找到List<SubscriberMethod>。

每个订阅者,相应一个List<SubscriberMethod>。有多少onEvent函数,返回的List,就有多少个item。

即查找订阅源内的事件处理方法,同一时候还会查到它的父类中的事件处理方法。返回list,交给

private void subscribe(Object subscriber, SubscriberMethod subscriberMethod, boolean sticky, int priority)

进行处理。

     Event与Subscriber之间,是一对多的关系。

即一个事件,能够被多个订阅者关注。

Subscriber与Event之间,也是一对多的关系。

即一个订阅者,能够订阅多个事件。

subscribe方法。也就是将上述的那样的关系。进行理顺,合理的建立map的映射关系,主要做了这样几件事件。

     a.依据SubscriberMethod中的EventType类型,将Subscribtion对象存放在subscriptionsByEventType中。建立EventType与Subscription的映射。一个事件能够有多个订阅者。

    b.依据Subscriber将EventType存放在typesBySubscriber`中,建立Subscriber到EventType的映射,每一个Subscriber能够订阅多个事件。

    c.假设是Sticky类型的订阅者,直接向它发送上个保存的事件(假设有的话)。

     通过Subscriber到EventType的映射,我们就能够非常方便地使一个Subscriber取消接收事件,通过EventType到Sucscribtion的映射,能够方便地将对应的事件发送到它的每个订阅者。

与Observer不同的是。使用EventBus。不同的被观察者,不需统一实现Observer中的interface方法。在上层代码中,也不须要逐一进行notify机制。通过Map进行订阅源与事件函数的相应关系,进行解耦,为其核心之处。

       发送流程:

         EventBus.getDefault().post(new EventType());參数为用户自己定义的对象。最为简单的处理方式,实现事件发送。

     当事件发送出去后,全部的订阅者。是怎样调用其事件方法的呢?这个就须要遍历上文提到的subscriptionsByEventType的Map了。Post发送事件,入口为post函数:public void post(Object event),在postSingleEvent函数中个,有一个重要的处理函数:

    /** Finds all Class objects including super classes and interfaces. */
private List<Class<?>> findEventTypes(Class<? > eventClass) {
synchronized (eventTypesCache) {
List<Class<?>> eventTypes = eventTypesCache.get(eventClass);
if (eventTypes == null) {
eventTypes = new ArrayList<Class<? >>();
Class<?> clazz = eventClass;
while (clazz != null) {
eventTypes.add(clazz);
addInterfaces(eventTypes, clazz.getInterfaces());
clazz = clazz.getSuperclass();
}
eventTypesCache.put(eventClass, eventTypes);
}
return eventTypes;
}
}

其作用。就是把这个事件类的对象、实现的接口及父类的类对象存到一个List中返回。依据list中的eventTypes,遍历subscriptionsByEventType,获取订阅源对象。进行逐一的调用事件函数。

        这里须要注意的是,当Post一个事件时,这个事件的父事件(事件类的父类事件)、接口事件也会被Post,所以假设订阅者接收Object类型的事件,即包括onEvent(Object object)事件函数。那么Subscriber就能够接收全部的事件。

          通过本篇博文的了解,EventBus就是通过Map。建立订阅源与事件函数的相应关系,进行解耦,来规避Observer的接口方法的多次、频繁的定义。

android平台中,EventBus研究学习的更多相关文章

  1. Android平台下Dalvik层hook框架ddi的研究

    通过adbi,可以对native层的所有代码进行hook.但对于Android系统来说,这还远远不够,因为很多应用都还是在Dalvik虚拟机中运行的. 那么,有没有什么办法可以对Dalvik虚拟机中跑 ...

  2. 【转】手把手教你读取Android版微信和手Q的聊天记录(仅作技术研究学习)

    1.引言 特别说明:本文内容仅用于即时通讯技术研究和学习之用,请勿用于非法用途.如本文内容有不妥之处,请联系JackJiang进行处理!   我司有关部门为了获取黑产群的动态,有同事潜伏在大量的黑产群 ...

  3. 手把手教你读取Android版微信和手Q的聊天记录(仅作技术研究学习)

    1.引言 特别说明:本文内容仅用于即时通讯技术研究和学习之用,请勿用于非法用途.如本文内容有不妥之处,请联系JackJiang进行处理!   我司有关部门为了获取黑产群的动态,有同事潜伏在大量的黑产群 ...

  4. Android消息传递之EventBus 3.0使用详解

    前言: 前面两篇不仅学习了子线程与UI主线程之间的通信方式,也学习了如何实现组件之间通信,基于前面的知识我们今天来分析一下EventBus是如何管理事件总线的,EventBus到底是不是最佳方案?学习 ...

  5. (转) unity 在移动平台中,文件操作路径详解

    http://www.unitymanual.com/thread-23491-1-1.html 今天,这篇文章其实是个老生常谈的问题咯,在网上类似的文章也比比皆是,在此我只是做个详细总结方便大家能够 ...

  6. 基于python的opcode优化和模块按需加载机制研究(学习与个人思路)(原创)

    基于python的opcode优化和模块按需加载机制研究(学习与思考) 姓名:XXX 学校信息:XXX 主用编程语言:python3.5 个人技术博客:http://www.cnblogs.com/M ...

  7. Android自复制传播APP原理学习(翻译)

     Android自复制传播APP原理学习(翻译) 1 背景介绍 论文链接:http://arxiv.org/abs/1511.00444 项目地址:https://github.com/Tribler ...

  8. unity 在移动平台中,文件操作路径详解

    今天,这篇文章其实是个老生常谈的问题咯,在网上类似的文章也比比皆是,在此我只是做个详细总结方便大家能够更好.更快的掌握,当然,如有不足的地方 欢迎指正!!! 相信大家在开发过程中,难免会保存一些文件在 ...

  9. Android中的SQLite使用学习

    Android中的SQLite使用学习 SQLite是非常流行的嵌入式关系型数据库,轻载, 速度快,而且是开源.在Android中,runtime提供SQLite,所以我们可以使用SQLite,而且是 ...

随机推荐

  1. 第一次PS练习

    嘿嘿,自己第一次的PS,虽然把在大神眼里是小KS,但是了,对我来说值得劲纪念.加油,我会努力的.

  2. sizeof与strlen的区别 浅谈

    1.sizeof operator sizeof是C语言的一种单目操作符,如C语言的其他操作符++.- - 等,它并不是函数. Queries size of the object or type. ...

  3. Bmp 解析 (2013-09-09 19:30:41)

    bmp 图片结构 位图格式(bmp)是显示图片的基本格式,其文件扩展名为*.BMP. 在Windows下,任何各式的图片文件(包括视频播放)都要转化为位图才能显示出来,其他各种格式的图片文件是在位图格 ...

  4. two sets of Qt binaries into the same process的解决办法

    突然出现了这样问题,吓死我,然后只是把原来编译好的app里面所有的东西删除再编译就好了. 如果删除后不行,可以试试后面的截图所说,反正我是没有试过的 Starting /Qtwork/build-te ...

  5. 2个NASM开发环境下载以及30篇教程(lostspeed)

    http://download.csdn.net/detail/lostspeed/8958175http://download.csdn.net/detail/lostspeed/8954263 h ...

  6. [置顶] Android中使用sqlite3操作SQLite

    SQLite库包含一个名字叫做sqlite3的命令行,它可以让用户手工输入并执行面向SQLite数据库的SQL命令.本文档提供一个样使用sqlite3的简要说明. 一.创建数据库:  1.将sqlit ...

  7. Android核心基础(十一)

    1.Android的状态栏通知(Notification) 通知用于在状态栏显示消息,消息到来时以图标方式表示,如下: //获取通知管理器 NotificationManager mNotificat ...

  8. PHP - 字符串操作

    第8章 字符串处理 学习要点: 1.字符串格式化 2.操作子字符串 3.字符串比较 4.查找替换字符串 5.处理中文字符 在每天的编程工作中,处理.调整以至最后控制字符串是很重要的一部分,一般也认为这 ...

  9. IIS - HTTP 错误 500.21 - Internal Server Error 处理程序“WebServiceHandlerFactory-Integrated”在其模块列表中有一个错误模块“ManagedPipelineHandler”

    http://www.cnblogs.com/yc-755909659/p/3445278.html 首先观察,aspnet_regiis.exe文件是不是损坏的,如果是,重新下载,覆盖即可,在百度云 ...

  10. ubuntu 14.04 使用极点五笔输入法

    相比12.04在外观改变不是非常大,但当中细节有些许变化,特别输入法非常不大好用,为此,我们使用fcitx输入法,使用我喜欢的五笔拼音,安装步骤例如以下: 方法一: 最新的方法非常easy: 安装14 ...