JetPack之ViewModel原理學(xué)習(xí)

JetPack的ViewModel的定位是用來存儲管理界面(Activity或Fragment)數(shù)據(jù)的類系羞,ViewModel中的數(shù)據(jù)可以由LiveData進(jìn)行存儲唧领。整個ViewModel的代碼不多救湖,總共就一百多行。主要學(xué)習(xí)一下ViewModel是如何結(jié)合kotlin的協(xié)程的使用,以及ViewModel是如何關(guān)聯(lián)Activity/Fragment的生命周期,在Activity/Fragment的生命周期結(jié)束之后做一些清理工作检访,防止內(nèi)存泄漏。

ViewModel中使用協(xié)程

androidx.lifecycle:lifecycle-viewModel-ktx在2.1.0之后提供了對ViewModel進(jìn)行了擴(kuò)展仔掸,提供了viewModelScope的擴(kuò)展屬性脆贵。因此我們可以在ViewModel中進(jìn)行一些協(xié)程操作。

private const val JOB_KEY = "androidx.lifecycle.ViewModelCoroutineScope.JOB_KEY"

val ViewModel.viewModelScope: CoroutineScope
        get() {
            val scope: CoroutineScope? = this.getTag(JOB_KEY)
            if (scope != null) {
                return scope
            }
            return setTagIfAbsent(JOB_KEY,
                CloseableCoroutineScope(SupervisorJob() + Dispatchers.Main.immediate))
        }

擴(kuò)展屬性復(fù)寫了get()方法起暮,主要是先會根據(jù)Key從ViewModel的mBagOfTags(一個Map)去取相應(yīng)的CoroutineScope卖氨,如果取不到再創(chuàng)建一個CloseableCoroutineScope并且放到ViewModel的mBagOfTags中。

internal class CloseableCoroutineScope(context: CoroutineContext) : Closeable, CoroutineScope {
    override val coroutineContext: CoroutineContext = context

    override fun close() {
        coroutineContext.cancel()
    }
}

CloseableCoroutineScope實現(xiàn)了Closeable接口,并且在close方法中對協(xié)程的Job調(diào)用cancel方法取消協(xié)程的Job筒捺。

如下是ViewModel中相關(guān)的代碼

  • mBagOfTags是一個HashMap柏腻,我們的viewModelScope屬性就是存儲在這個Map中
  • 當(dāng)使用viewModelScope不存在的時候,我們會嘗試創(chuàng)建一個CloseableCoroutineScope的對象系吭,并且放入到mBagOfTags這個Map中
  • viewModelScope是一個CloseableCoroutineScope的對象五嫂,這個類實現(xiàn)了Closeable接口,并且在close方法中對協(xié)程的Job進(jìn)行了cancel
  • 在ViewModel的clear()方法中會調(diào)用mBagOfTags中所有Closeable對象的close()方法
public abstract class ViewModel {
    private final Map<String, Object> mBagOfTags = new HashMap<>();
    private volatile boolean mCleared = false;

    @MainThread
    final void clear() {
        mCleared = true;
        if (mBagOfTags != null) {
            synchronized (mBagOfTags) {
            //遍歷Map中的所有Closeable對象肯尺,調(diào)用close方法沃缘,防止內(nèi)存泄漏
                for (Object value : mBagOfTags.values()) {
                    closeWithRuntimeException(value);
                }
            }
        }
        onCleared();
    }

    @SuppressWarnings("unchecked")
    <T> T setTagIfAbsent(String key, T newValue) {
        T previous;
        synchronized (mBagOfTags) {
            previous = (T) mBagOfTags.get(key);
            if (previous == null) {
                mBagOfTags.put(key, newValue);
            }
        }
        T result = previous == null ? newValue : previous;
        if (mCleared) {
            closeWithRuntimeException(result);
        }
        return result;
    }

    <T> T getTag(String key) {
        if (mBagOfTags == null) {
            return null;
        }
        synchronized (mBagOfTags) {
            return (T) mBagOfTags.get(key);
        }
    }

    private static void closeWithRuntimeException(Object obj) {
        if (obj instanceof Closeable) {
            try {
                ((Closeable) obj).close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

ViewModel關(guān)聯(lián)組件生命周期

在上面ViewModel的源碼中可以看到很多防止內(nèi)存泄漏的操作都是在ViewModel的clear()方法進(jìn)行,順著這個思路可以倒過來溯源则吟,clear()方法是在哪里被調(diào)用了槐臀。往上追溯發(fā)現(xiàn)是在ViewModelStore的clear()方法中調(diào)用了。

// Class to store {@code ViewModels}.
public class ViewModelStore {
    private final HashMap<String, ViewModel> mMap = new HashMap<>();
    
    /**
     *  Clears internal storage and notifies ViewModels that they are no longer used.
     */
    public final void clear() {
        for (ViewModel vm : mMap.values()) {
            vm.clear();
        }
        mMap.clear();
    }
}

根據(jù)注釋ViewModelStore這個類是用來存儲ViewModel的逾滥,ViewModel會被放在HashMap中峰档。那ViewModelStore的clear()方法又是在哪里調(diào)用呢?這里分析一下Activity的情況寨昙。

public ComponentActivity() {
       getLifecycle().addObserver(new LifecycleEventObserver() {
            @Override
            public void onStateChanged(@NonNull LifecycleOwner source,
                    @NonNull Lifecycle.Event event) {
                if (event == Lifecycle.Event.ON_DESTROY) {
                    if (!isChangingConfigurations()) {
                        getViewModelStore().clear();
                    }
                }
            }
        });
}

在androidx中的ComponentActivity會去監(jiān)聽Lifecycle的生命周期的變化讥巡,在生命周期是ON_DESTROY的時候回去調(diào)用ViewModelStore的的clear()方法。

整個流程如下:
Activity生命周期走到ON_DESTROY -> 調(diào)用ViewModelStore#clear()方法 -> 遍歷ViewModelStore中所有的ViewModel舔哪,調(diào)用ViewModel#clear()方法 -> 調(diào)用ViewModel中的所有Closeable對象的close方法&回調(diào)onCleared()方法欢顷。

tips:平時需要在ViewModel中做一些清除/反注冊的操作的時候可以復(fù)寫onCleared()方法,在這個方法中做操作

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末捉蚤,一起剝皮案震驚了整個濱河市抬驴,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌缆巧,老刑警劉巖布持,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異陕悬,居然都是意外死亡题暖,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門捉超,熙熙樓的掌柜王于貴愁眉苦臉地迎上來胧卤,“玉大人,你說我怎么就攤上這事拼岳≈μ埽” “怎么了?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵惜纸,是天一觀的道長叶撒。 經(jīng)常有香客問我绝骚,道長,這世上最難降的妖魔是什么痊乾? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任皮壁,我火速辦了婚禮,結(jié)果婚禮上哪审,老公的妹妹穿的比我還像新娘蛾魄。我一直安慰自己,他們只是感情好湿滓,可當(dāng)我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布滴须。 她就那樣靜靜地躺著,像睡著了一般叽奥。 火紅的嫁衣襯著肌膚如雪扔水。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天朝氓,我揣著相機(jī)與錄音魔市,去河邊找鬼。 笑死赵哲,一個胖子當(dāng)著我的面吹牛待德,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播枫夺,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼将宪,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了橡庞?” 一聲冷哼從身側(cè)響起较坛,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎扒最,沒想到半個月后丑勤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡吧趣,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年确封,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片再菊。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖颜曾,靈堂內(nèi)的尸體忽然破棺而出纠拔,到底是詐尸還是另有隱情,我是刑警寧澤泛豪,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布稠诲,位于F島的核電站侦鹏,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏臀叙。R本人自食惡果不足惜略水,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望劝萤。 院中可真熱鬧渊涝,春花似錦、人聲如沸床嫌。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽厌处。三九已至鳖谈,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間阔涉,已是汗流浹背缆娃。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留瑰排,地道東北人贯要。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像凶伙,于是被迫代替她去往敵國和親郭毕。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,713評論 2 354