日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Android ViewModel的使用總結

瀏覽:196日期:2022-09-17 18:13:31
目錄基本使用MainRepositoryMainViewModelMainActivity

ViewModel 相關問題是高頻面試題。主要源于它是 MVVM 架構模式的重要組件,并且它可以在因配置更改導致頁面銷毀重建時依然保留 ViewModel 實例。

看看 ViewModel 的生命周期

Android ViewModel的使用總結

ViewModel 只有在正常 Activity finish 時才會被清除。

問題來了:

為什么Activity旋轉屏幕后ViewModel可以恢復數據 ViewModel 的實例緩存到哪兒了 什么時候 ViewModel#onCleared() 會被調用

在解決這三個問題之前,回顧下 ViewModel 的用法特性

基本使用MainRepository

class MainRepository { suspend fun getNameList(): List<String> {return withContext(Dispatchers.IO) { listOf('張三', '李四')} }}MainViewModel

class MainViewModel: ViewModel() { private val nameList = MutableLiveData<List<String>>() val nameListResult: LiveData<List<String>> = nameList private val mainRepository = MainRepository()fun getNames() {viewModelScope.launch { nameList.value = mainRepository.getNameList()} }}MainActivity

class MainActivity : AppCompatActivity() { // 創建 ViewModel 方式 1 // 通過 kotlin 委托特性創建 ViewModel // 需添加依賴 implementation ’androidx.activity:activity-ktx:1.2.3’ // viewModels() 內部也是通過 創建 ViewModel 方式 2 來創建的 ViewModel private val mainViewModel: MainViewModel by viewModels() override fun onCreate(savedInstanceState: Bundle?) {super.onCreate (savedInstanceState)setContentView(R.layout.activity_main)// 創建 ViewModel 方式 2val mainViewModel = ViewModelProvider(this).get(MainViewModel::class.java)mainViewModel.nameListResult.observe(this, { Log.i('MainActivity', 'mainViewModel: nameListResult: $it') Log.i('MainActivity', 'MainActivity: ${this@MainActivity} mainViewModel: $mainViewModel mainViewModel.nameListResult: ${mainViewModel.nameListResult}')})mainViewModel.getNames() }}

測試步驟:打開app -> 正常看到日志

18:03:02.575 : mainViewModel: nameListResult: [張三, 李四]18:03:02.575 : com.yqy.myapplication.MainActivity@7ffa77 mainViewModel: com.yqy.myapplication.MainViewModel@29c0057 mainViewModel.nameListResult: androidx.lifecycle.MutableLiveData@ed0d744

接著測試步驟:打開設置更換系統語言 -> 切換到當前app所在的任務 再看日志

18:03:59.622 : mainViewModel: nameListResult: [張三, 李四]18:03:59.622 : com.yqy.myapplication.MainActivity@49a4455 mainViewModel: com.yqy.myapplication.MainViewModel@29c0057 mainViewModel.nameListResult: androidx.lifecycle.MutableLiveData@ed0d744

神奇!MainActivity 被重建了,而 ViewModel 的實例沒有變,并且 ViewModel 對象里的 LiveData 對象實例也沒變。 這就是 ViewModel 的特性。

ViewModel 出現之前,Activity 可以使用 onSaveInstanceState() 方法保存,然后從 onCreate() 中的 Bundle 恢復數據,但此方法僅適合可以序列化再反序列化的少量數據(IPC 對 Bundle 有 1M 的限制),而不適合數量可能較大的數據,如用戶信息列表或位圖。 ViewModel 的出現完美解決這個問題。

我們先看看 ViewModel 怎么創建的: 通過上面的實例代碼,最終 ViewModel 的創建方法是

val mainViewModel = ViewModelProvider(this).get(MainViewModel::class.java)

神奇!MainActivity 被重建了,而 ViewModel 的實例沒有變,并且 ViewModel 對象里的 LiveData 對象實例也沒變。 這就是 ViewModel 的特性。

ViewModel 出現之前,Activity 可以使用 onSaveInstanceState() 方法保存,然后從 onCreate() 中的 Bundle 恢復數據,但此方法僅適合可以序列化再反序列化的少量數據(IPC 對 Bundle 有 1M 的限制),而不適合數量可能較大的數據,如用戶信息列表或位圖。 ViewModel 的出現完美解決這個問題。

我們先看看 ViewModel 怎么創建的: 通過上面的實例代碼,最終 ViewModel 的創建方法是

val mainViewModel = ViewModelProvider(this).get(MainViewModel::class.java)

創建 ViewModelProvider 對象并傳入了 this 參數,然后通過 ViewModelProvider#get 方法,傳入 MainViewModel 的 class 類型,然后拿到了 mainViewModel 實例。

ViewModelProvider 的構造方法

public ViewModelProvider(@NonNull ViewModelStoreOwner owner) { // 獲取 owner 對象的 ViewModelStore 對象this(owner.getViewModelStore(), owner instanceof HasDefaultViewModelProviderFactory? ((HasDefaultViewModelProviderFactory) owner).getDefaultViewModelProviderFactory(): NewInstanceFactory.getInstance()); }

ViewModelProvider 構造方法的參數類型是 ViewModelStoreOwner ?ViewModelStoreOwner 是什么?我們明明傳入的 MainActivity 對象呀! 看看 MainActivity 的父類們發現

public class ComponentActivity extends androidx.core.app.ComponentActivity implements...// 實現了 ViewModelStoreOwner 接口ViewModelStoreOwner,...{ private ViewModelStore mViewModelStore;// 重寫了 ViewModelStoreOwner 接口的唯一的方法 getViewModelStore() @NonNull @Override public ViewModelStore getViewModelStore() {if (getApplication() == null) { throw new IllegalStateException('Your activity is not yet attached to the ' + 'Application instance. You can’t request ViewModel before onCreate call.');}ensureViewModelStore();return mViewModelStore; }

ComponentActivity 類實現了 ViewModelStoreOwner 接口。 奧 ~~ 剛剛的問題解決了。

再看看剛剛的 ViewModelProvider 構造方法里調用了 this(ViewModelStore, Factory),將 ComponentActivity#getViewModelStore 返回的 ViewModelStore 實例傳了進去,并緩存到 ViewModelProvider 中

public ViewModelProvider(@NonNull ViewModelStore store, @NonNull Factory factory) {mFactory = factory;// 緩存 ViewModelStore 對象mViewModelStore = store; }

接著看 ViewModelProvider#get 方法做了什么

@MainThread public <T extends ViewModel> T get(@NonNull Class<T> modelClass) {String canonicalName = modelClass.getCanonicalName();if (canonicalName == null) { throw new IllegalArgumentException('Local and anonymous classes can not be ViewModels');}return get(DEFAULT_KEY + ':' + canonicalName, modelClass); }

獲取 ViewModel 的 CanonicalName , 調用了另一個 get 方法

@MainThread public <T extends ViewModel> T get(@NonNull String key, @NonNull Class<T> modelClass) { // 從 mViewModelStore 緩存中嘗試獲取ViewModel viewModel = mViewModelStore.get(key);// 命中緩存if (modelClass.isInstance(viewModel)) { if (mFactory instanceof OnRequeryFactory) {((OnRequeryFactory) mFactory).onRequery(viewModel); } // 返回緩存的 ViewModel 對象 return (T) viewModel;} else { //noinspection StatementWithEmptyBody if (viewModel != null) {// TODO: log a warning. }}// 使用工廠模式創建 ViewModel 實例if (mFactory instanceof KeyedFactory) { viewModel = ((KeyedFactory) mFactory).create(key, modelClass);} else { viewModel = mFactory.create(modelClass);}// 將創建的 ViewModel 實例放進 mViewModelStore 緩存中mViewModelStore.put(key, viewModel);// 返回新創建的 ViewModel 實例return (T) viewModel; }

mViewModelStore 是啥?通過 ViewModelProvider 的構造方法知道 mViewModelStore 其實是我們 Activity 里的 mViewModelStore 對象,它在 ComponentActivity 中被聲明。 看到了 put 方法,不難猜它內部用了 Map 結構。

public class ViewModelStore {// 果不其然,內部有一個 HashMap private final HashMap<String, ViewModel> mMap = new HashMap<>(); final void put(String key, ViewModel viewModel) {ViewModel oldViewModel = mMap.put(key, viewModel);if (oldViewModel != null) { oldViewModel.onCleared();} }// 通過 key 獲取 ViewModel 對象 final ViewModel get(String key) {return mMap.get(key); } Set<String> keys() {return new HashSet<>(mMap.keySet()); } /** * 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(); }}

到這兒正常情況下 ViewModel 的創建流程看完了,似乎沒有解決任何問題~ 簡單總結:ViewModel 對象存在了 ComponentActivity 的 mViewModelStore 對象中。 第二個問題解決了:ViewModel 的實例緩存到哪兒了

轉換思路 mViewModelStore 出現頻率這么高,何不看看它是什么時候被創建的呢?

記不記得剛才看 ViewModelProvider 的構造方法時 ,獲取 ViewModelStore 對象時,實際調用了 MainActivity#getViewModelStore() ,而 getViewModelStore() 實現在 MainActivity 的父類 ComponentActivity 中。

// ComponentActivity#getViewModelStore() @Override public ViewModelStore getViewModelStore() {if (getApplication() == null) { throw new IllegalStateException('Your activity is not yet attached to the ' + 'Application instance. You can’t request ViewModel before onCreate call.');}ensureViewModelStore();return mViewModelStore; }

在返回 mViewModelStore 對象之前調用了 ensureViewModelStore()

void ensureViewModelStore() {if (mViewModelStore == null) { NonConfigurationInstances nc = (NonConfigurationInstances) getLastNonConfigurationInstance(); if (nc != null) {// Restore the ViewModelStore from NonConfigurationInstancesmViewModelStore = nc.viewModelStore; } if (mViewModelStore == null) {mViewModelStore = new ViewModelStore(); }} }

當 mViewModelStore == null 調用了 getLastNonConfigurationInstance() 獲取 NonConfigurationInstances 對象 nc,當 nc != null 時將 mViewModelStore 賦值為 nc.viewModelStore,最終 viewModelStore == null 時,才會創建 ViewModelStore 實例。

不難發現,之前創建的 viewModelStore 對象被緩存在 NonConfigurationInstances 中

// 它是 ComponentActivity 的靜態內部類 static final class NonConfigurationInstances {Object custom;// 果然在這兒ViewModelStore viewModelStore; }

當 mViewModelStore == null 調用了 getLastNonConfigurationInstance() 獲取 NonConfigurationInstances 對象 nc,當 nc != null 時將 mViewModelStore 賦值為 nc.viewModelStore,最終 viewModelStore == null 時,才會創建 ViewModelStore 實例。

不難發現,之前創建的 viewModelStore 對象被緩存在 NonConfigurationInstances 中

// 它是 ComponentActivity 的靜態內部類 static final class NonConfigurationInstances {Object custom;// 果然在這兒ViewModelStore viewModelStore; }

NonConfigurationInstances 對象通過 getLastNonConfigurationInstance() 來獲取的

// Activity#getLastNonConfigurationInstance /** * Retrieve the non-configuration instance data that was previously * returned by {@link #onRetainNonConfigurationInstance()}. This will * be available from the initial {@link #onCreate} and * {@link #onStart} calls to the new instance, allowing you to extract * any useful dynamic state from the previous instance. * * <p>Note that the data you retrieve here should <em>only</em> be used * as an optimization for handling configuration changes. You should always * be able to handle getting a null pointer back, and an activity must * still be able to restore itself to its previous state (through the * normal {@link #onSaveInstanceState(Bundle)} mechanism) even if this * function returns null. * * <p><strong>Note:</strong> For most cases you should use the {@link Fragment} API * {@link Fragment#setRetainInstance(boolean)} instead; this is also * available on older platforms through the Android support libraries. * * @return the object previously returned by {@link #onRetainNonConfigurationInstance()} */ @Nullable public Object getLastNonConfigurationInstance() {return mLastNonConfigurationInstances != null? mLastNonConfigurationInstances.activity : null; }

好長一段注釋,大概意思有幾點:

onRetainNonConfigurationInstance 方法和 getLastNonConfigurationInstance 是成對出現的,跟 onSaveInstanceState 機制類似,只不過它是僅用作處理配置更改的優化。 返回的是 onRetainNonConfigurationInstance 返回的對象

onRetainNonConfigurationInstance 和 getLastNonConfigurationInstance 的調用時機在本篇文章不做贅述,后續文章會進行解釋。

看看 onRetainNonConfigurationInstance 方法

/*** 保留所有適當的非配置狀態*/ @Override @Nullable @SuppressWarnings('deprecation') public final Object onRetainNonConfigurationInstance() {// Maintain backward compatibility.Object custom = onRetainCustomNonConfigurationInstance();ViewModelStore viewModelStore = mViewModelStore;// 若 viewModelStore 為空,則嘗試從 getLastNonConfigurationInstance() 中獲取if (viewModelStore == null) { // No one called getViewModelStore(), so see if there was an existing // ViewModelStore from our last NonConfigurationInstance NonConfigurationInstances nc = (NonConfigurationInstances) getLastNonConfigurationInstance(); if (nc != null) {viewModelStore = nc.viewModelStore; }}// 依然為空,說明沒有需要緩存的,則返回 nullif (viewModelStore == null && custom == null) { return null;}// 創建 NonConfigurationInstances 對象,并賦值 viewModelStoreNonConfigurationInstances nci = new NonConfigurationInstances();nci.custom = custom;nci.viewModelStore = viewModelStore;return nci; }

到這兒我們大概明白了,Activity 在因配置更改而銷毀重建過程中會先調用 onRetainNonConfigurationInstance 保存 viewModelStore 實例。 在重建后可以通過 getLastNonConfigurationInstance 方法獲取之前的 viewModelStore 實例。

現在解決了第一個問題:為什么Activity旋轉屏幕后ViewModel可以恢復數據

再看第三個問題:什么時候 ViewModel#onCleared() 會被調用

public abstract class ViewModel { protected void onCleared() { } @MainThread final void clear() {mCleared = true;// Since clear() is final, this method is still called on mock objects// and in those cases, mBagOfTags is null. It’ll always be empty though// because setTagIfAbsent and getTag are not final so we can skip// clearing itif (mBagOfTags != null) { synchronized (mBagOfTags) {for (Object value : mBagOfTags.values()) { // see comment for the similar call in setTagIfAbsent closeWithRuntimeException(value);} }}onCleared(); }}

onCleared() 方法被 clear() 調用了。 剛才看 ViewModelStore 源碼時好像是調用了 clear() ,回顧一下:

public class ViewModelStore { private final HashMap<String, ViewModel> mMap = new HashMap<>(); final void put(String key, ViewModel viewModel) {ViewModel oldViewModel = mMap.put(key, viewModel);if (oldViewModel != null) { oldViewModel.onCleared();} } final ViewModel get(String key) {return mMap.get(key); } Set<String> keys() {return new HashSet<>(mMap.keySet()); } /** * 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(); }}

onCleared() 方法被 clear() 調用了。 剛才看 ViewModelStore 源碼時好像是調用了 clear() ,回顧一下:

public class ViewModelStore { private final HashMap<String, ViewModel> mMap = new HashMap<>(); final void put(String key, ViewModel viewModel) {ViewModel oldViewModel = mMap.put(key, viewModel);if (oldViewModel != null) { oldViewModel.onCleared();} } final ViewModel get(String key) {return mMap.get(key); } Set<String> keys() {return new HashSet<>(mMap.keySet()); } /** * 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(); }}

在 ViewModelStore 的 clear() 中,遍歷 mMap 并調用 ViewModel 對象的 clear() , 再看 ViewModelStore 的 clear() 什么時候被調用的:

// ComponentActivity 的構造方法 public ComponentActivity() {... getLifecycle().addObserver(new LifecycleEventObserver() { @Override public void onStateChanged(@NonNull LifecycleOwner source, @NonNull Lifecycle.Event event) {if (event == Lifecycle.Event.ON_DESTROY) { // Clear out the available context mContextAwareHelper.clearAvailableContext(); // And clear the ViewModelStore if (!isChangingConfigurations()) {getViewModelStore().clear(); }} }});... }

觀察當前 activity 生命周期,當 Lifecycle.Event == Lifecycle.Event.ON_DESTROY,并且 isChangingConfigurations() 返回 false 時才會調用 ViewModelStore#clear 。

// Activity#isChangingConfigurations() /** * Check to see whether this activity is in the process of being destroyed in order to be * recreated with a new configuration. This is often used in * {@link #onStop} to determine whether the state needs to be cleaned up or will be passed * on to the next instance of the activity via {@link #onRetainNonConfigurationInstance()}. * * @return If the activity is being torn down in order to be recreated with a new configuration, * returns true; else returns false. */ public boolean isChangingConfigurations() {return mChangingConfigurations; }

isChangingConfigurations 用來檢測當前的 Activity 是否因為 Configuration 的改變被銷毀了, 配置改變返回 true,非配置改變返回 false。

總結,在 activity 銷毀時,判斷如果是非配置改變導致的銷毀, getViewModelStore().clear() 才會被調用。

第三個問題:什么時候 ViewModel#onCleared() 會被調用 解決!

以上就是Android ViewModel的使用總結的詳細內容,更多關于Android ViewModel的使用的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产美女高潮在线观看| 日韩a一区二区| 激情婷婷综合| 国产一区亚洲| 黄色av一区| 伊人久久婷婷| 亚洲精品在线二区| 日韩1区2区日韩1区2区| 综合国产精品| 国产图片一区| 美女视频免费精品| 精品国产欧美日韩| 日韩.com| 日韩啪啪电影网| 99久久婷婷| 久久亚洲风情| 日本在线视频一区二区| 国产调教一区二区三区| 精品一区视频| 在线日韩电影| 亚洲精品女人| 国产精品日韩精品在线播放| 精品国产欧美日韩| 麻豆mv在线观看| 国产综合亚洲精品一区二| 香蕉精品999视频一区二区| 亚洲三级网站| 精品一区二区三区四区五区| 欧美日韩国产v| 日韩在线观看一区二区| 国产精品白丝一区二区三区| 国产+成+人+亚洲欧洲在线| 亚洲性图久久| 亚洲久久视频| 欧美成人一二区| 九九精品调教| 亚州精品视频| 黄色网一区二区| 999精品一区| 日本aⅴ亚洲精品中文乱码| 国产精品福利在线观看播放| 亚洲作爱视频| 国产精品久久久亚洲一区| 久久久夜精品| 国产欧美成人| 黄色免费成人| 久久精品国产一区二区| 欧美特黄一区| 久久久久97| 免费精品视频| 精品午夜视频| 久色成人在线| а√在线中文在线新版| 国产综合精品| 欧美黄色一区二区| 99国产精品| 久久精品国产在热久久| 午夜在线精品偷拍| 国产成人黄色| 日韩精品91亚洲二区在线观看| 人在线成免费视频| 国产欧美三级| 亚洲欧美网站| 久久久久久自在自线| 亚洲精品三级| 中文字幕在线免费观看视频| 日本在线不卡视频| 99久久亚洲精品| 国产精区一区二区| 日韩一级不卡| 狠狠久久伊人| 日本成人一区二区| 亚洲视频国产| 久久午夜精品| 亚洲一区二区三区四区五区午夜 | 视频在线观看国产精品| 免费福利视频一区二区三区| 国产精品久久久久久模特| 日韩一区精品| 在线综合亚洲| 亚洲国产日韩欧美在线| 国内亚洲精品| 亚洲特级毛片| 激情六月综合| 午夜精品亚洲| 亚洲欧美一区在线| 99精品电影| 日韩精品欧美| 亚洲一级特黄| 亚洲精品一区二区妖精| 欧美日韩国产欧| 亚洲免费激情| 视频一区二区中文字幕| 蜜桃久久av一区| 三级亚洲高清视频| 蘑菇福利视频一区播放| 蜜桃视频免费观看一区| 亚洲免费资源| 日本99精品| 国产日产一区| 精品午夜视频| 色偷偷偷在线视频播放| 成人在线网站| 亚洲激情另类| 一区二区三区四区在线观看国产日韩 | 国产综合精品一区| 欧美一级二区| 四虎国产精品免费久久| 丝袜美腿一区二区三区| 91久久在线| 亚洲激情二区| 中国女人久久久| 尹人成人综合网| 欧美日韩国产在线观看网站 | 欧美永久精品| 欧美日韩1区| 欧美亚洲人成在线| 男人的天堂亚洲一区| 欧美专区18| 中文视频一区| 日韩精品久久理论片| 国产精品亚洲一区二区在线观看| 国产精品久久久久久模特| 精品国产一级| 香蕉久久99| 午夜亚洲福利| 精品久久久亚洲| 在线国产一区| 欧美亚洲tv| 电影亚洲精品噜噜在线观看| 亚洲永久字幕| 欧美日韩夜夜| 国产成人免费| 国产精品美女久久久浪潮软件| 91精品国产自产精品男人的天堂| 国产在线观看91一区二区三区| 欧美日韩国产免费观看视频| 日本va欧美va瓶| 手机在线电影一区| 久久不射网站| 久久99久久人婷婷精品综合| 亚洲高清激情| 青草av.久久免费一区| 成人在线视频区| 欧美日韩国产欧| 国产精品亚洲综合色区韩国| 日韩精品一区二区三区免费观看| 亚洲欧美日韩国产| 久久成人福利| 精品91久久久久| 欧美交a欧美精品喷水| 国产二区精品| 国产精品调教| 欧美~级网站不卡| 国产美女久久| 99xxxx成人网| 麻豆国产精品视频| 夜夜精品视频| 精品视频在线你懂得| 丝袜美腿亚洲一区二区图片| 精品福利久久久| 蜜臀av国产精品久久久久 | 国产成人精品一区二区三区免费| 欧美专区在线| 日韩精品中文字幕第1页| 日韩福利视频导航| 国产精品99免费看| 国产精品66| 亚洲欧美日本视频在线观看| 国产成人免费| 日韩国产欧美视频| 亚洲男女av一区二区| 精品久久视频| 亚洲精品三级| 国产一区清纯| 国产一区福利| 亚洲影院天堂中文av色| 久久精品二区三区| 另类欧美日韩国产在线| 亚洲免费成人av在线| 欧美日韩在线二区| 精品一区二区三区免费看 | 欧美特黄一区| 91日韩在线| 国产伦精品一区二区三区视频| 国产亚洲在线观看| 樱桃视频成人在线观看| 91福利精品在线观看| 91久久黄色| 亚洲精品一区三区三区在线观看| 国产精品羞羞答答在线观看| 免费日韩av片| 欧美中文一区二区| 国产一区一一区高清不卡| 91综合久久爱com| 免费成人av在线播放| 99国产精品免费视频观看| 精品免费av| 国产欧美啪啪| 日韩精品一区二区三区中文在线 |