欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android?ViewModel創(chuàng)建不受橫豎屏切換影響原理詳解

 更新時(shí)間:2023年03月08日 09:45:28   作者:大強(qiáng)Dev  
這篇文章主要為大家介紹了Android?ViewModel創(chuàng)建不受橫豎屏切換影響原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

ViewModel的創(chuàng)建方式

在我們項(xiàng)目中, 引入了viewModel 做MVI 設(shè)計(jì)模式的組成部分,它是JetPack 組件庫(kù)中的重要成員。今天來(lái)了解下它。

// 在 Activity 中使用
class MainActivity : AppCompatActivity() {
   // 使用 Activity 的作用域
   private val viewModel : MainViewModel by viewModels()
}
// 在 Fragment 中使用
class MainFragment : Fragment() {
   // 使用 Activity 的作用域,與 MainActivity 使用同一個(gè)對(duì)象
   val activityViewModel : MainViewModel by activityViewModels()
   // 使用 Fragment 的作用域
   val viewModel : MainViewModel by viewModels()
}
// ViewModel 創(chuàng)建工廠
private final Factory mFactory;
// ViewModel 存儲(chǔ)容器
private final ViewModelStore mViewModelStore;
// 默認(rèn)使用 NewInstanceFactory 反射創(chuàng)建 ViewModel
public ViewModelProvider(ViewModelStoreOwner owner) {
    this(owner.getViewModelStore(), ... NewInstanceFactory.getInstance());
}
// 自定義 ViewModel 創(chuàng)建工廠
public ViewModelProvider(ViewModelStoreOwner owner, Factory factory) {
    this(owner.getViewModelStore(), factory);
}
// 記錄宿主的 ViewModelStore 和 ViewModel 工廠
public ViewModelProvider(ViewModelStore store, Factory factory) {
    mFactory = factory;
    mViewModelStore = store;
}
@NonNull
@MainThread
public <T extends ViewModel> T get(Class<T> modelClass) {
    String canonicalName = modelClass.getCanonicalName();
    if (canonicalName == null) {
        throw new IllegalArgumentException("Local and anonymous classes can not be ViewModels");
    }
    // 使用類名作為緩存的 KEY
    return get(DEFAULT_KEY + ":" + canonicalName, modelClass);
}
// Fragment
@NonNull
@MainThread
public <T extends ViewModel> T get(String key, Class<T> modelClass) {
    // 1. 先從 ViewModelStore 中取緩存
    ViewModel viewModel = mViewModelStore.get(key);
    if (modelClass.isInstance(viewModel)) {
        return (T) viewModel;
    }
    // 2. 使用 ViewModel 工廠創(chuàng)建實(shí)例
    viewModel = mFactory.create(modelClass);
    ...
    // 3. 存儲(chǔ)到 ViewModelStore
    mViewModelStore.put(key, viewModel);
    return (T) viewModel;
}
// 默認(rèn)的 ViewModel 工廠
public static class NewInstanceFactory implements Factory {
    private static NewInstanceFactory sInstance;
    @NonNull
    static NewInstanceFactory getInstance() {
        if (sInstance == null) {
            sInstance = new NewInstanceFactory();
        }
        return sInstance;
    }
    @NonNull
    @Override
    public <T extends ViewModel> T create(Class<T> modelClass) {
        // 反射創(chuàng)建 ViewModel 對(duì)象
        return modelClass.newInstance();
    }
}

可以看到:

ViewModel 實(shí)例的方法最終是通過(guò) ViewModelProvider 完成的。ViewModelProvider 可以理解為創(chuàng)建 ViewModel 的工具類,它需要 2 個(gè)參數(shù):

參數(shù) 1 ViewModelStoreOwner:

它對(duì)應(yīng)于 Activity / Fragment 等持有 ViewModel 的宿主,它們內(nèi)部通過(guò) ViewModelStore 維持一個(gè) ViewModel 的映射表,ViewModelStore 是實(shí)現(xiàn) ViewModel 作用域和數(shù)據(jù)恢復(fù)的關(guān)鍵;

參數(shù) 2 Factory:

它對(duì)應(yīng)于 ViewModel 的創(chuàng)建工廠,缺省時(shí)將使用默認(rèn)的 NewInstanceFactory 工廠來(lái)反射創(chuàng)建 ViewModel 實(shí)例。

創(chuàng)建 ViewModelProvider 工具類后,你將通過(guò) get() 方法來(lái)創(chuàng)建 ViewModel 的實(shí)例。get() 方法內(nèi)部首先會(huì)通過(guò) ViewModel 的全限定類名從映射表(ViewModelStore)中取緩存,未命中才會(huì)通過(guò) ViewModel 工廠創(chuàng)建實(shí)例再緩存到映射表中。 正因?yàn)橥粋€(gè) ViewModel 宿主使用的是同一個(gè) ViewModelStore 映射表,因此在同一個(gè)宿主上重復(fù)調(diào)用 ViewModelProvider.get() 返回同一個(gè) ViewModel 實(shí)例。

// ViewModel 本質(zhì)上就是一個(gè)映射表而已
public class ViewModelStore {
   // <String - ViewModel> 哈希表
   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());
   }
   public final void clear() {
       for (ViewModel vm : mMap.values()) {
           vm.clear();
       }
       mMap.clear();
   }
}

ViewModel 宿主是 ViewModelStoreOwner 接口的實(shí)現(xiàn)類,例如 ComponentActivity:

public class ComponentActivity extends androidx.core.app.ComponentActivity implements
    ContextAware,
    LifecycleOwner,
    ViewModelStoreOwner ... {
    // ViewModel 的存儲(chǔ)容器
    private ViewModelStore mViewModelStore;
    // ViewModel 的創(chuàng)建工廠
    private ViewModelProvider.Factory mDefaultFactory;
    @NonNull
    @Override
    public ViewModelStore getViewModelStore() {
        if (mViewModelStore == null) {
            // 已簡(jiǎn)化過(guò)程
            mViewModelStore = new ViewModelStore();
        }
        return mViewModelStore;
    }
}

由此,ViewModel 的創(chuàng)建其實(shí)跟activity 是相關(guān)聯(lián)的。

ViewModel 為什么不受 Activity 橫豎屏生命周期的影響

1、在 Activity 走到 onDestroy 方法時(shí),做了判斷 isChangingConfigurations

可以看到,在 ComponentActivity 構(gòu)造方法中添加了生命周期的判斷,當(dāng) Activity onDestroy 時(shí),如果是發(fā)生了橫豎屏切換,就不會(huì)走 getViewModelStore().clear(),清理操作,保證了ViewModel 持有的數(shù)據(jù)還能存在。

2、在 Activity 獲取 getViewModelStore 時(shí),

通過(guò)getLastNonConfigurationInstance() 獲取 NonConfigurationInstances 實(shí)例,從而得到這個(gè)實(shí)例中的 viewModelStore

而且,Activity 生命周期的變化都會(huì)走這個(gè)方法,來(lái)保證viewModelStore 不為空。

3、onRetainNonConfigurationInstance 調(diào)用

在 Activity 屏幕旋轉(zhuǎn)時(shí),onRetainNonConfigurationInstance()onStoponDestroy之間調(diào)用

onRetainNonConfigurationInstance() 是個(gè)重要的方法

這保證了 在銷毀前,viewModelStore 實(shí)例被拿到并交給 NonConfigurationInstances 實(shí)例,將 viewModelStore 賦值給他

總結(jié)

1、介紹了ViewModel的創(chuàng)建

  • ViewModel 由創(chuàng)建工廠 Factory mFactory 反射創(chuàng)建而來(lái),并被放到存儲(chǔ)容器 ViewModelStore 中

2、ViewModel生命周期和橫豎屏場(chǎng)景下不被銷毀的原因

在 Activity 走到 onDestroy 方法時(shí),做了判斷 isChangingConfigurations,如果發(fā)生橫豎屏,不會(huì)清理ViewModelStore,所以ViewModel還在

銷毀前 onRetainNonConfigurationInstance() 被觸發(fā),將原來(lái)的viewModelStore 賦值給 NonConfigurationInstances實(shí)例。

從Activity重建后 調(diào)用 getViewModelStore方法,內(nèi)部 通過(guò) getLastNonConfigurationInstance() 獲取 NonConfigurationInstances , NonConfigurationInstances 中保存的viewModelStore 實(shí)例被拿到,而viewModelStore 中保存著 ViewModel 。

以上就是Android ViewModel創(chuàng)建不受橫豎屏切換影響原理詳解的詳細(xì)內(nèi)容,更多關(guān)于Android ViewModel創(chuàng)建的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

最新評(píng)論