LiveDataBus

LiveDataBus是基于LiveData实现的类似EventBus的消息通信框架,它是基于LiveData实现的,完全可以代替EventBus,RxBus;

往期通信框架的缺点:
  • EventBus :原理实现复杂,无法混淆,需要手动绑定生命周期
  • Handler : 容易导致内存泄漏,空指针,高耦合,不利于维护
  • RxBus:依赖于RxJava,包太大,影响apk大小,app启动时间
  • interface:实现复杂,写多了不利于维护
LiveDataBus的优点:
  • 无需引入依赖,使用官方提供的LiveData类
  • 自动感应组件的生命周期,无内存泄漏风险
  • 代码量少,能完全替代EventBus

发布订阅模式

LiveDataBus的主要是基于发布订阅设计模式,发布订阅模式定义了一种 “一对多” 的关系,和观察者模式是完全不同的两个设计模式;

举个例子
  • 观察者模式:
    观察者,被观察者,订阅关系,被观察者通知观察者,观察者做出反应。出租车司机(被观察者)到目的地通知乘客(观察者)下车;
  • 发布订阅模式:
    发布者,订阅者,订阅关系,订阅者监听发布者的变化,发布者会通知所有的订阅者,使他们做出反应。火车到站了乘务员(发布者)喊:南京站到了,通知所有的(旅客)订阅者,旅客(订阅者)判断自己是否到南京站,到南京的旅客下车(回调);

LiveData

下面详细介绍核心类LiveData

LiveData是一个可以被观察的数据holder,并且可以自动感知控件的生命周期,不会发生内存泄漏;

LiveData需要一个观察者对象,当LiveData的值发生改变时,观察者会察觉到这个改变;

liveData.observe(this, new Observer<String>() {
            @Override
            public void onChanged(@Nullable String s) {

            }
        });

使用livedata注册观察者监听

mutableLiveData.observe(this, new Observer() {
            @Override
            public void onChanged(@Nullable Object o) {
                
            }
        });

使用livedata发送消息给观察者

 mutableLiveData.postValue("发送的消息");

LiveData其实就是一个存放数据的holder,类似ViewHolder的holder,存放在LiveData里的数据会拥有LiveData的特性;

LiveData的特性:

  • UI和实时数据保持一致,因为LiveData采用的是观察者模式,这样一来就可以在数据发生改变时获得通知,更新UI。
  • 避免内存泄漏,观察者被绑定到组件的生命周期上,当被绑定的组件销毁(destroy)时,观察者会立刻自动清理自身的数据。
  • 不会再产生由于Activity处于stop状态而引起的崩溃,例如:当Activity处于后台状态时,是不会收到LiveData的任何事件的。
  • 不需要再解决生命周期带来的问题,LiveData可以感知被绑定的组件的生命周期,只有在活跃状态才会通知数据变化。
  • 实时数据刷新,当组件处于活跃状态或者从不活跃状态到活跃状态时总是能收到最新的数据。
  • 解决Configuration Change问题,在屏幕发生旋转或者被回收再次启动,立刻就能收到最新的数据。

LiveData是Android Architecture Components的一个类;这个类是谷歌在Google I/O 2017发布一套帮助开发者解决Android架构设计的方案。这个类有四个核心,后续会一一介绍;

  • Room,一个强大的SQLite对象映射库。
  • LiveData,一个可感知生命周期、可被观察的数据容器,它可以存储数据,还会在数据发生改变时进行提醒。
  • Lifecycle,包含LifeCycleOwer和LifecycleObserver,分别是生命周期所有者和生命周期感知者。
  • ViewMoudle,一类对象,它用于为UI组件提供数据,在设备配置发生变更时依旧可以存活。

LifeDataBus的原理


LifeDataBus的组成:基于发布订阅模式

  • 消息:any和object
  • 消息通道:LiveData扮演了消息通道的角色,不同的消息通道用不同的名字区分,名字是String类型的,可以通过名字获取到一个LiveData消息通道。
  • 消息总线:消息总线通过单例实现,不同的消息通道存放在一个HashMap中。
  • 订阅者:订阅者通过getChannel获取消息通道,然后调用observe订阅这个通道的消息。
  • 发布者:发布者通过getChannel获取消息通道,然后调用setValue或者postValue发布消息。

LiveDataBus的第一代实现

public class LiveDataBus {
    
    // 消息总线
    private Map<String,MutableLiveData<Object>> bus;

    // 单例模式(静态内部类法)
    private LiveDataBus(){
        bus = new HashMap<>();
    }
    private static class SingleInstance{
       private static LiveDataBus mInstance = new LiveDataBus();
    }
    public static LiveDataBus get(){
        return SingleInstance.mInstance;
    }

    // 获取消息通道
    public<T> MutableLiveData<T> getChanel(String target,Class<T> type){
        if (!bus.containsKey(target)){
            bus.put(target, new MediatorLiveData<>());
        }
        return (MutableLiveData<T>) bus.get(target);
    }
    public MutableLiveData<Object> getChanel(String target){
        return getChanel(target,Object.class);
    }
    
}
我们使用单例确保只有一个消息总线(Map),在消息总线中存放消息通道(LiveData),通过map的key去匹配消息通道两端的发送者和订阅者。

用第一代LiveDataBus订阅

// 注册LiveDataBus
        LiveDataBus.get().getChanel("yingyingying").observe(this, new android.arch.lifecycle.Observer<Object>() {
            @Override
            public void onChanged(@Nullable Object o) {
                Toast.makeText(getApplicationContext(),""+o.toString(),Toast.LENGTH_LONG).show();
            }
        });

发送消息

tv_hello.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                LiveDataBus.get().getChanel("yingyingying",String.class).postValue("嘤嘤嘤");
            }
        });
接收到消息

至于说他是第一代bus,说明肯定有问题,问题就是在post或者set一个value后,只要在一个frag/act里observe了,无论组件是否启动,都会收到value,即当在act1中post了,在act2中observe,但是post的时候act2没有运行,当启动act2,收到了value。收到了订阅前的消息

分析原因——LiveData的原理

通过查看LiveData的源码发现setValue()开始,依次调用了
dispatchValue() -> considerNotify() -> observer.onChanged()

protected void setValue(T value) {
        assertMainThread("setValue");
        mVersion++;
        mData = value;
        dispatchingValue(null);
    }
private void dispatchingValue(@Nullable ObserverWrapper initiator) {
        if (mDispatchingValue) {
            mDispatchInvalidated = true;
            return;
        }
        mDispatchingValue = true;
        do {
            mDispatchInvalidated = false;
            if (initiator != null) {
                considerNotify(initiator);
                initiator = null;
            } else {
                for (Iterator<Map.Entry<Observer<T>, ObserverWrapper>> iterator =
                        mObservers.iteratorWithAdditions(); iterator.hasNext(); ) {
                    considerNotify(iterator.next().getValue());
                    if (mDispatchInvalidated) {
                        break;
                    }
                }
            }
        } while (mDispatchInvalidated);
        mDispatchingValue = false;
    }
private void considerNotify(ObserverWrapper observer) {
        if (!observer.mActive) {
            return;
        }
        // Check latest state b4 dispatch. Maybe it changed state but we didn't get the event yet.
        //
        // we still first check observer.active to keep it as the entrance for events. So even if
        // the observer moved to an active state, if we've not received that event, we better not
        // notify for a more predictable notification order.
        if (!observer.shouldBeActive()) {
            observer.activeStateChanged(false);
            return;
        }
        if (observer.mLastVersion >= mVersion) {
            return;
        }
        observer.mLastVersion = mVersion;
        //noinspection unchecked
        observer.mObserver.onChanged((T) mData);
    }

postValue()会调用setValue()所以同理;
这就解释了为什么我们可以在observer中收到post来的value,为什么act2不运行也可以收到value;
我们注意到当observer.mLastVersion >= mVersion的时候会直接return,不调用onChanged从而解决上面的问题。

  • mLastVersion :这是observer的一个描述版本的变量;


    1

    2

    全文只有这两个地方对此变量赋值

  • mVersion:这是LiveData的版本变量;


    1

    2

    3

    全文有三个地方,用到此变量;

  • mVersion是LiveData的版本,初始为-1,每次setValue(发送消息)的时候++

  • mLastVersion 是observer的版本,初始为-1,当调用onChanged的时候赋值为mVersion
    也就是说对于每一个observer,它的初始值都是-1,当observe的版本小于livedata时,会强制回调onChanged(),然后将observer的版本赋值为livedata的版本(mVersion,此时相等,不会再回调onChanged)

根据上面的分析,我们只需要在订阅的时候,将observe的版本等于livedata的版本即可,这样livedata就不会强制调用onChanged();

我们需要拿到mLastVersion,就需要拿到observer对象,顺着源码发现observer对象存在mObservers的map中;我们自定义一个mutableLiveData,改写他的observe(),在observe()中,通过反射拿到mObservers对象,从而拿到observer.mLastVersion,将mVersion赋值给他;

LiveDataBus2.0

public class LiveDataBus2 {

    private final Map<String, BusMutableLiveData<Object>> bus;

    private LiveDataBus2() {
        bus = new HashMap<>();
    }

    private static class SingletonHolder {
        private static final LiveDataBus2 DEFAULT_BUS = new LiveDataBus2();
    }

    public static LiveDataBus2 get() {
        return SingletonHolder.DEFAULT_BUS;
    }

    public <T> MutableLiveData<T> with(String key, Class<T> type) {
        if (!bus.containsKey(key)) {
            bus.put(key, new BusMutableLiveData<>());
        }
        return (MutableLiveData<T>) bus.get(key);
    }

    public MutableLiveData<Object> with(String key) {
        return with(key, Object.class);
    }

    private static class ObserverWrapper<T> implements Observer<T> {

        private Observer<T> observer;

        public ObserverWrapper(Observer<T> observer) {
            this.observer = observer;
        }

        @Override
        public void onChanged(@Nullable T t) {
            if (observer != null) {
                if (isCallOnObserve()) {
                    return;
                }
                observer.onChanged(t);
            }
        }

        private boolean isCallOnObserve() {
            StackTraceElement[] stackTrace = Thread.currentThread().getStackTrace();
            if (stackTrace != null && stackTrace.length > 0) {
                for (StackTraceElement element : stackTrace) {
                    if ("android.arch.lifecycle.LiveData".equals(element.getClassName()) &&
                            "observeForever".equals(element.getMethodName())) {
                        return true;
                    }
                }
            }
            return false;
        }
    }

    private static class BusMutableLiveData<T> extends MutableLiveData<T> {

        private Map<Observer, Observer> observerMap = new HashMap<>();

        @Override
        public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<T> observer) {
            super.observe(owner, observer);
            try {
                // 设置observer的version和LiveData一致
                hook(observer);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

        @Override
        public void observeForever(@NonNull Observer<T> observer) {
            if (!observerMap.containsKey(observer)) {
                observerMap.put(observer, new ObserverWrapper(observer));
            }
            super.observeForever(observerMap.get(observer));
        }

        @Override
        public void removeObserver(@NonNull Observer<T> observer) {
            Observer realObserver = null;
            if (observerMap.containsKey(observer)) {
                realObserver = observerMap.remove(observer);
            } else {
                realObserver = observer;
            }
            super.removeObserver(realObserver);
        }

        private void hook(@NonNull Observer<T> observer) throws Exception {
            // 获取livedata的class对象
            Class<LiveData> classLiveData = LiveData.class;
            // 获取   LiveData类的mObservers对象 (Map对象)的 Field对象
            Field fieldObservers = classLiveData.getDeclaredField("mObservers");
            // 将mObservers 的private设置为 public
            fieldObservers.setAccessible(true);
            //  获取当前livedata的mObservers对象(map)
            Object objectObservers = fieldObservers.get(this);
            // 拿到mObservers(map)的class对象
            Class<?> classObservers = objectObservers.getClass();
            // 通过map的class对象拿到get()的method对象
            Method methodGet = classObservers.getDeclaredMethod("get", Object.class);
            methodGet.setAccessible(true);
            // 通过map 的 get Method对象 拿到值 (Entry)  (arg1:map ,arg2:key )
            Object objectWrapperEntry = methodGet.invoke(objectObservers, observer);
            // 拿到wrapper
            Object objectWrapper = null;
            if (objectWrapperEntry instanceof Map.Entry) {
                objectWrapper = ((Map.Entry) objectWrapperEntry).getValue();
            }
            if (objectWrapper == null) {
                throw new NullPointerException("Wrapper can not be bull!");
            }
            // 反射wrapper对象
            Class<?> classObserverWrapper = objectWrapper.getClass().getSuperclass();
            // 拿到wrapper的version
            Field fieldLastVersion = classObserverWrapper.getDeclaredField("mLastVersion");
            fieldLastVersion.setAccessible(true);
            //get livedata's version
            Field fieldVersion = classLiveData.getDeclaredField("mVersion");
            fieldVersion.setAccessible(true);
            Object objectVersion = fieldVersion.get(this);
            //set wrapper's version
            fieldLastVersion.set(objectWrapper, objectVersion);
        }
    }
}

hook的作用相当于 在observe()调用后执行observer.mLastVersion = mVersion; 让considerNotify()直接return,可是我们如何收到订阅后的post呢?因为只有订阅的时候才会hook,在hook后,我们调用post(),会mVersion++,所以在判断 if (observer.mLastVersion >= mVersion) 的时候就又会是false了;

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

推荐阅读更多精彩内容

  • Day 63 释义:easy to talk to or deal with 例句:My English tea...
    叶小静Stamy阅读 389评论 0 0
  • 希逋阅读 204评论 0 0
  • 2018年又一场大雪,与2008年的大雪不相上下,雪还是那样的雪,而时隔十年的我却变得不再一样! 十年前的我,全部...
    木木夕阳无限好阅读 219评论 0 0
  • 望不到尽头的生活,该怎么过?我这样问自己。 可是过了好久,我都没有回答自己。因为我突然想到,是什么样的生活,被我定...
    叮咚的你阅读 1,058评论 1 2
  • 春风扫去千山雪, 细雨送来万顷蓝。 林海苍茫大漠隐, 草原牧女羊群闲。 祁连北引燕然水, 南贯昆仑青藏原。 不尽黄...
    云逸1108阅读 177评论 0 1