JetPacks之?dāng)?shù)據(jù)傳遞工具

相比RxBus猜极,EventBus,LiveData有個(gè)非常簡(jiǎn)單的LiveEventBus
參考文章
類(lèi)似github 項(xiàng)目

如果了解JetPacks原理需要查看我的文章
JetPacks之Lifecycles 原理分析

我們自定義分析下原理

貼一下我們的自定義工具類(lèi)

public class LiveDataBus {
//存放訂閱者
private Map<String, MutableLiveData<Object>> bus;
private static LiveDataBus liveDataBus = new LiveDataBus();

private LiveDataBus() {
    bus = new HashMap();
}
public static LiveDataBus getInstance() {
    return liveDataBus;
}
//注冊(cè)訂閱者
public synchronized <T> MutableLiveData<T> with(String key, Class<T> type) {
    if(!bus.containsKey(key)){
        bus.put(key,new MutableLiveData<Object>());
    }
    return (MutableLiveData<T>)bus.get(key);
}
}

實(shí)驗(yàn)

做一個(gè)頁(yè)面跳轉(zhuǎn)椭符,第一個(gè)頁(yè)面?zhèn)鬟f數(shù)據(jù)給第二個(gè)頁(yè)面
5s發(fā)送一個(gè)數(shù)據(jù)。分別看下LiveDataBus 和LiveDataBusX「接續(xù)往下看會(huì)實(shí)現(xiàn)這個(gè)」粘性數(shù)據(jù)

-> 第一個(gè)頁(yè)面
public void startLiveDataBusActivity() {
    //-> LiveDataBus 換成 LiveDataBusX 解決粘性
    //-> 粘性數(shù)據(jù)執(zhí)行
    //-> new LiveData() ——> 綁定Observer —>  setValue(onChanged)    正常LiveDataBusX
    //-> new LiveData() ——> setValue(onChanged) —> 綁定Observer     LiveDataBus
    Intent intent = new Intent();
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    intent.setClass(context, TestLiveDataBusActivity.class);
    context.startActivity(intent);
    new Thread() {
        @Override
        public void run() {
            for (int i = 0; i < 10; i++) {
                //發(fā)送消息
                LiveDataBus.getInstance().with("data", String.class).postValue("David-LiveDataBus");
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            }
        }
    }.start();
}

我們?cè)赥estLiveDataBusActivity 這個(gè)頁(yè)面進(jìn)行接收

public class TestLiveDataBusActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_test_live_data_bus);

    //-> 測(cè)試 要吧MainActivity LiveDataBus和LiveDataBusX 保持一致
    LiveDataBus.getInstance().with("data",String.class)
            .observe(this, new Observer<String>() {
                @Override
                public void onChanged(String s) {
                    if(s!=null)
                        Toast.makeText(TestLiveDataBusActivity.this, s, Toast.LENGTH_SHORT).show();
                }
            });
}
}

我們會(huì)發(fā)現(xiàn)可能某些場(chǎng)景我們不適合用

我們正常的思維

new LiveData() ——> 綁定Observer —> setValue(onChanged)

先綁定在設(shè)置值

而這里忽略我們的順序變成
new LiveData() ——> setValue(onChanged) —> 綁定Observer -> setValue(onChanged)

和LiveData原理實(shí)現(xiàn)有關(guān), 我們可以用反射切斷第一次的onChange

解決這個(gè)bug耻姥,用反射處理

-> 「找Hook 點(diǎn)」

再次我們閱讀源碼

找到我們的入口
//發(fā)送消息
LiveDataBus.getInstance().with("data", String.class).postValue("David-LiveDataBus");
兜了一圈销钝,你會(huì)發(fā)現(xiàn),會(huì)切換線程琐簇,并且給指向setValue方法蒸健。

@MainThread
protected void setValue(T value) {
    assertMainThread("setValue");
    mVersion++;
    mData = value;
    -> 繼續(xù)進(jìn)入查看
    dispatchingValue(null);
}

需要注意-----------》 我們dipatchingValue傳入的一開(kāi)始null

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<? super T>, ObserverWrapper>> iterator =
                    mObservers.iteratorWithAdditions(); iterator.hasNext(); ) {
                -> 初始值null參數(shù),所以我們會(huì)走for循環(huán)
                considerNotify(iterator.next().getValue());
                if (mDispatchInvalidated) {
                    break;
                }
            }
        }
    } while (mDispatchInvalidated);
    mDispatchingValue = false;
}

considerNotify ---------> 這里我們發(fā)現(xiàn)了private實(shí)現(xiàn)婉商,并且是靠version判斷似忧。貌似是Hook的切入點(diǎn)

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);
}

從這個(gè)方法我們知道改變mLastVersion 和mVersion的值就可以了。mLastVersion 初始化值 -1据某。執(zhí)行過(guò)程不一定橡娄。但是確實(shí)mLastVersion < mVersion 導(dǎo)致繼續(xù)向下執(zhí)行

我們需要從這里反向退,最好反向看找到最終的調(diào)用點(diǎn)

-> 這個(gè)是我們推出來(lái)的反射點(diǎn)
SafeIterableMap<Observer<? super T>, ObserverWrapper> mObservers

//ObserverWrapper initiator-> 發(fā)現(xiàn)迭代mObservers

void dispatchingValue(@Nullable ObserverWrapper initiator)

->  considerNotify(iterator.next().getValue());

private void considerNotify(ObserverWrapper observer) {

observer.mLastVersion

-> 反射實(shí)現(xiàn)代碼

public class LiveDataBusX {
//存放訂閱者
private Map<String, BusMutableLiveData<Object>> bus;

private static LiveDataBusX liveDataBus = new LiveDataBusX();

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

public static LiveDataBusX getInstance() {
    return liveDataBus;
}

//注冊(cè)訂閱者癣籽,(存入map) Hook前用MutableLiveData
public synchronized <T> BusMutableLiveData<T> with(String key, Class<T> type) {
    if (!bus.containsKey(key)) {
        bus.put(key, new BusMutableLiveData<Object>());
    }
    return (BusMutableLiveData<T>) bus.get(key);
}

public static class BusMutableLiveData<T> extends MutableLiveData<T> {
    @Override
    public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
        super.observe(owner, observer);
        hook(observer);
    }

    //在這里去改變onChange的流程
    private void hook(Observer<? super T> observer) {
        try {
            //1.得到mLastVersion
            //獲取到LiveData的類(lèi)中的mObservers對(duì)象
            Class<LiveData> liveDataClass = LiveData.class;
            Field mObserversField = liveDataClass.getDeclaredField("mObservers");
            mObserversField.setAccessible(true);
            //獲取到這個(gè)成員變量的對(duì)象
            Object mObserversObject = mObserversField.get(this);
            //得到map對(duì)應(yīng)的class對(duì)象
            Class<?> mObserversClass = mObserversObject.getClass();
            //獲取到mObservers對(duì)象的get方法
            Method get = mObserversClass.getDeclaredMethod("get", Object.class);
            get.setAccessible(true);
            //執(zhí)行g(shù)et方法
            Object invokeEntry = get.invoke(mObserversObject, observer);
            //定義一個(gè)空的對(duì)象
            Object observerWraper = null;
            if (invokeEntry != null && invokeEntry instanceof Map.Entry) {
                observerWraper = ((Map.Entry) invokeEntry).getValue();
            }
            if (observerWraper == null) {
                throw new NullPointerException("observerWraper is null");
            }
            //得到ObserverWrapper的類(lèi)對(duì)象  編譯擦除問(wèn)題會(huì)引起多態(tài)沖突所以用getSuperclass
            Class<?> superclass = observerWraper.getClass().getSuperclass();
            Field mLastVersion = superclass.getDeclaredField("mLastVersion");
            mLastVersion.setAccessible(true);
            //2.得到mVersion
            Field mVersion = liveDataClass.getDeclaredField("mVersion");
            mVersion.setAccessible(true);
            //3.把mVersion的數(shù)據(jù)填入到mLastVersion中
            Object mVersionValue = mVersion.get(this);
            mLastVersion.set(observerWraper, mVersionValue);


        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
禁止轉(zhuǎn)載挽唉,如需轉(zhuǎn)載請(qǐng)通過(guò)簡(jiǎn)信或評(píng)論聯(lián)系作者。
  • 序言:七十年代末筷狼,一起剝皮案震驚了整個(gè)濱河市瓶籽,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌埂材,老刑警劉巖塑顺,帶你破解...
    沈念sama閱讀 218,284評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡严拒,警方通過(guò)查閱死者的電腦和手機(jī)扬绪,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)裤唠,“玉大人挤牛,你說(shuō)我怎么就攤上這事≈终海” “怎么了墓赴?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,614評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)航瞭。 經(jīng)常有香客問(wèn)我诫硕,道長(zhǎng),這世上最難降的妖魔是什么刊侯? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,671評(píng)論 1 293
  • 正文 為了忘掉前任章办,我火速辦了婚禮,結(jié)果婚禮上滔吠,老公的妹妹穿的比我還像新娘纲菌。我一直安慰自己,他們只是感情好疮绷,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,699評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布翰舌。 她就那樣靜靜地躺著,像睡著了一般冬骚。 火紅的嫁衣襯著肌膚如雪椅贱。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,562評(píng)論 1 305
  • 那天只冻,我揣著相機(jī)與錄音庇麦,去河邊找鬼。 笑死喜德,一個(gè)胖子當(dāng)著我的面吹牛山橄,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播舍悯,決...
    沈念sama閱讀 40,309評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼航棱,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了萌衬?” 一聲冷哼從身側(cè)響起饮醇,我...
    開(kāi)封第一講書(shū)人閱讀 39,223評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎秕豫,沒(méi)想到半個(gè)月后朴艰,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,668評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,859評(píng)論 3 336
  • 正文 我和宋清朗相戀三年祠墅,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了侮穿。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,981評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡饵隙,死狀恐怖撮珠,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情金矛,我是刑警寧澤,帶...
    沈念sama閱讀 35,705評(píng)論 5 347
  • 正文 年R本政府宣布勺届,位于F島的核電站驶俊,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏免姿。R本人自食惡果不足惜饼酿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,310評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望胚膊。 院中可真熱鬧故俐,春花似錦、人聲如沸紊婉。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,904評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)喻犁。三九已至槽片,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間肢础,已是汗流浹背还栓。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,023評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留传轰,地道東北人剩盒。 一個(gè)月前我還...
    沈念sama閱讀 48,146評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像慨蛙,于是被迫代替她去往敵國(guó)和親辽聊。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,933評(píng)論 2 355