Android跨界面共享數(shù)據(jù)——LiveData應(yīng)用

業(yè)務(wù)場(chǎng)景

ezgif.com-video-to-gif.gif

3個(gè)界面中有3個(gè)獨(dú)立控件,需要同步他們的狀態(tài)告抄,即其中任一控件狀態(tài)變化撰茎,其余兩個(gè)隨之而變。

解決方案


1. 傳遞值:startActivityForResult() + onActivityResult()

這是最容易想到的方案玄妈,實(shí)現(xiàn)步驟如下:

  • 在界面A將控件狀態(tài)封裝在Intent
  • 在界面A通過(guò)startActivityForResult()跳轉(zhuǎn)到界面B
  • 在界面A返回之前通過(guò)setResult()將控件狀態(tài)返回給界面A
  • 在界面A的onActivityResult()中獲取控件狀態(tài)并更新UI

但該方案有缺點(diǎn):

  1. 代碼可讀性較差乾吻,特別是當(dāng)onActivityResult()中還夾雜著其他業(yè)務(wù)邏輯。
  2. 增加了Activity間的耦合(即Activiy B依賴于Activity A的特殊傳值方式拟蜻,Activity A依賴于Activity B的回傳值)绎签。因?yàn)榻缑骈g是兩兩耦合的,所以也導(dǎo)致了擴(kuò)展性較差酝锅,如果需求改成“從Activity A直接跳轉(zhuǎn)到Activity B”诡必,需要重新出處理Activity AActivity B的跳轉(zhuǎn)邏輯。

2. 共享值(持久化)

既然通過(guò)傳遞值的方式不夠好,那直接“共享值”呢爸舒?即將每次狀態(tài)改變都持久化(存在本地)蟋字,每次繪制界面都從本地讀取狀態(tài)。

設(shè)想界面A中有一個(gè)列表扭勉,每個(gè)表項(xiàng)都包含一個(gè)需要狀態(tài)同步的控件鹊奖,當(dāng)服務(wù)器返回一批新數(shù)據(jù)后,需要挨個(gè)將數(shù)據(jù)進(jìn)行存儲(chǔ)涂炎,隨著列表不斷刷新忠聚,本地存儲(chǔ)的內(nèi)容就不斷增多,為控制本地存儲(chǔ)占用的空間唱捣,在 App 退出時(shí)需清空本地存儲(chǔ)两蟀。

3. 共享值(LiveData)

既然在 App 退出時(shí)需要清空數(shù)據(jù),則表明控件狀態(tài)信息的生命周期和 App 的生命周期同步震缭,而持久化解決的問(wèn)題是生命周期長(zhǎng)于 App 生命周期的情況赂毯。于是第三個(gè)解決方案就閃亮登場(chǎng)了~~~

LiveData是谷歌在Google I/O 2017發(fā)布的Android Architecture Components(Google教你如何寫 App 系列)中的一項(xiàng)內(nèi)容。

對(duì)于當(dāng)前這個(gè)case拣宰,LiveData充當(dāng)如下角色:

  • LiveData是一個(gè)數(shù)據(jù)持有者党涕,但不像一般的數(shù)據(jù)持有者,它可以感知系統(tǒng)組件的生命周期徐裸。
  • LiveData可以被觀察遣鼓,但它不像一般的觀察者模式(一有數(shù)據(jù)變動(dòng)就通知所有觀察者)。只有當(dāng)被觀察者處于激活狀態(tài)時(shí)才被通知重贺。

所以基于LiveData的解決方案如下:將控件狀態(tài)信息保存在LiveData中骑祟,三個(gè)不同的界面分別觀察LiveData

通過(guò)觀察者模式將方案1中數(shù)據(jù)傳遞問(wèn)題轉(zhuǎn)換為數(shù)據(jù)共享气笙,三個(gè)界面沒(méi)有絲毫耦合次企。將LiveData設(shè)置為單例,使其和 App 生命周期相一致潜圃,也避免了開辟額外的本地存儲(chǔ)缸棵。

LiveData應(yīng)用


1. 創(chuàng)建狀態(tài)信息實(shí)體類

將要共享的狀態(tài)信息封裝成實(shí)體類,簡(jiǎn)單起見(jiàn)谭期,demo將狀態(tài)信息設(shè)置為int值堵第,如下:

public class Status {
    private int level;

    public int getLevel() {
        return level;
    }

    public void setLevel(int level) {
        this.level = level;
    }
}

2. 創(chuàng)建LiveData單例

下面的代碼只是將狀態(tài)信息實(shí)體類和LiveData關(guān)聯(lián),并將LiveData定義為單例隧出,方便跨界面使用踏志。

public class StatusLiveData extends MutableLiveData<Status> {
    private StatusLiveData() {
    }

    private static class Holder {
        public static final StatusLiveData INSTANCE = new StatusLiveData();
    }

    public static StatusLiveData getInstance() {
        return Holder.INSTANCE;
    }
}

//MutableLiveData在LiveData基礎(chǔ)上暴露兩個(gè)設(shè)值接口
public class MutableLiveData<T> extends LiveData<T> {
    @Override
    public void postValue(T value) {
        super.postValue(value);
    }

    @Override
    public void setValue(T value) {
        super.setValue(value);
    }
}

3. 為L(zhǎng)iveData添加觀察者

LiveData的觀察者通常是帶有生命周期概念的組件,比如Activity胀瞪,F(xiàn)ragment等等针余。觀察者需實(shí)現(xiàn)Observer<T>接口,以定義數(shù)據(jù)變化時(shí)做出的響應(yīng)。

public class ActivityA extends AppCompatActivity implements View.OnClickListener, Observer<Status> {
    private int level;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        ...
        StatusLiveData.getInstance().observe(this, this);
    }
    
    ...
    
    @Override
    public void onChanged(@Nullable Status status) {
        /**
         * get status data when it is changed and update UI
         */
        int level = status.getLevel();
        changeArrowStatus(level);
    }
}

4. 更新LiveData

最后一步就是在狀態(tài)值變化時(shí)候調(diào)用LiveData.setValue()更新數(shù)據(jù)圆雁。這里的邏輯和具體業(yè)務(wù)相關(guān)忍级,demo中的業(yè)務(wù)場(chǎng)景是點(diǎn)擊ImageView控件時(shí)改變其圖片。

public class ActivityB extends AppCompatActivity implements View.OnClickListener, Observer<Status> {
    private int level;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ...
        findViewById(R.id.iv_arrow).setOnClickListener(this);
        ...
    }

    @Override
    public void onClick(View v) {
        int id = v.getId();
        switch (id) {
            ...
            case R.id.iv_arrow:
                changeArrowStatus(++level);
                putStatus(level);
                break;
        }
    }

    /**
     * put status data into LiveData when data is changed
     */
    private void putStatus(int level) {
        Status status = new Status();
        status.setLevel(level);
        StatusLiveData.getInstance().setValue(status);
    }

    private void changeArrowStatus(int level) {
        ImageView ivArrow = findViewById(R.id.iv_arrow);
        LevelListDrawable levelListDrawable = ((LevelListDrawable) ivArrow.getDrawable());
        levelListDrawable.setLevel(level % 2);
    }
}

talk is cheap, show me the code

拋磚引玉伪朽,若大家有更好的方案轴咱,歡迎交流~~

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市驱负,隨后出現(xiàn)的幾起案子嗦玖,更是在濱河造成了極大的恐慌患雇,老刑警劉巖跃脊,帶你破解...
    沈念sama閱讀 217,826評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異苛吱,居然都是意外死亡酪术,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門翠储,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)绘雁,“玉大人,你說(shuō)我怎么就攤上這事援所÷郏” “怎么了?”我有些...
    開封第一講書人閱讀 164,234評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵住拭,是天一觀的道長(zhǎng)挪略。 經(jīng)常有香客問(wèn)我,道長(zhǎng)滔岳,這世上最難降的妖魔是什么杠娱? 我笑而不...
    開封第一講書人閱讀 58,562評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮谱煤,結(jié)果婚禮上摊求,老公的妹妹穿的比我還像新娘。我一直安慰自己刘离,他們只是感情好室叉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著硫惕,像睡著了一般茧痕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上疲憋,一...
    開封第一講書人閱讀 51,482評(píng)論 1 302
  • 那天凿渊,我揣著相機(jī)與錄音,去河邊找鬼。 笑死埃脏,一個(gè)胖子當(dāng)著我的面吹牛搪锣,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播彩掐,決...
    沈念sama閱讀 40,271評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼构舟,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了堵幽?” 一聲冷哼從身側(cè)響起狗超,我...
    開封第一講書人閱讀 39,166評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎朴下,沒(méi)想到半個(gè)月后努咐,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,608評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡殴胧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評(píng)論 3 336
  • 正文 我和宋清朗相戀三年渗稍,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片团滥。...
    茶點(diǎn)故事閱讀 39,926評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡竿屹,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出灸姊,到底是詐尸還是另有隱情拱燃,我是刑警寧澤,帶...
    沈念sama閱讀 35,644評(píng)論 5 346
  • 正文 年R本政府宣布力惯,位于F島的核電站碗誉,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏夯膀。R本人自食惡果不足惜诗充,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望诱建。 院中可真熱鬧蝴蜓,春花似錦、人聲如沸俺猿。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)押袍。三九已至诵冒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間谊惭,已是汗流浹背汽馋。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工侮东, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人豹芯。 一個(gè)月前我還...
    沈念sama閱讀 48,063評(píng)論 3 370
  • 正文 我出身青樓悄雅,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親铁蹈。 傳聞我的和親對(duì)象是個(gè)殘疾皇子宽闲,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容