ViewModel 类被设计用来管理 UI 相关数据,持有一个或多个 LiveData。
它的特性:生命周期不随系统配置改变的影响,组件被进程杀掉后,支持数据恢复。
ViewModel 创建流程
基本流程:ViewModelProvider 类的 get() 方法,首先,根据 key ,从 ViewModelStore 中查找,如果不存在,利用内部 ViewModelProvider.Factory 工厂创建,并存入 ViewModelStore。
ViewModelStore 存储类是为了 ViewModel 的生命周期保持。工厂可配置,系统默认提供几种工厂。
ViewModelProvider 的构造方法有组件 Owner 参数,如下。
public ViewModelProvider(@NonNull ViewModelStoreOwner owner) {
this(owner.getViewModelStore(), owner instanceof HasDefaultViewModelProviderFactory
? ((HasDefaultViewModelProviderFactory) owner).getDefaultViewModelProviderFactory()
: NewInstanceFactory.getInstance());
}
如果组件同时实现以上接口,(例ComponentActivity) ,从组件中拿 ViewModelStore 和 Factory ,ComponentActivity 内部默认工厂是 SavedStateViewModelFactory,它创建的 ViewModel 支持数据恢复。
ViewModel 创建时,不会传入 Context ,如果需要 Context ,创建 AndroidViewModel 。
配置改变的生命周期保持
系统配置改变时流程
onRetainNonConfigurationInstance() 方法,在 ActivityThread类 performDestoryActivity() 触发,此时,已经执行过 onSaveInstanceState(),还未执行onDestory()。
该方法用于将组件中的 ViewModelStore 对象,保存在NonConfigurationInstances,返回该 NonConfigurationInstances,即保存状态。
系统组件重建时,在 onCreate()方法,会通过 getViewModelStore() 构建 内部 ViewModelStore 对象,优先 getLastNonConfigurationInstance() 方法 恢复原有状态 (同一个对象),从NonConfigurationInstances中获取ViewModelStore。
因此,在配置改变时,ViewModel 对象是不变的。
只有当组件真正结束时,ViewModel 的生命周期才会结束,由系统执行 onClear() 方法。
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();
}
}
}
});
这是 ComponentActivity 基类组件 构造方法中注册的一个生命周期Observer,当组件销毁时,检测是否因 Configuration 改变导致销毁,主动销毁才会通过 ViewModelStore 清理ViewModel, onClear() 方法。
总结:组件利用 onRetainNonConfigurationInstance() 和 getLastNonConfigurationInstance() ,实现组件内部 ViewModelStore存储器 的 保存和恢复。
屏幕旋转,可以通过恢复的 ViewModelStore 获取 ViewModel ,其数据是不需要重新获取。
组件被系统回收时 ViewModel 数据恢复
本质是利用 onCreate() 和 onSaveInstanceState() 方法。
支持需要工厂类创建 ViewModel 时,构造方法参数包含SavedStateHandle,如 SavedStateViewModelFactory 工厂、或抽象 AbstractSavedStateViewModelFactory 工厂。
该类本身是一Map数据,内部一个SavedStateProvider,将Map数据→Bundle。
自动注册到SavedStateRegistry。
两种场景:
1,组件恢复数据时,在 onCreate() 方法,读取 Bundle ,通过内部控制器 SavedStateRegistryController 委派给 SavedStateRegistry。
将 Bundle mRestoredState 保存在 SavedStateRegistry。
ViewModel 创建时,消费掉该 Bundle,转存入 SavedStateHandle 的 Map 。
2,组件回收时,基础组件 ComponentActivity 的 onSaveInstanceState() 方法,通过内部控制器 SavedStateRegistryController 委派给 SavedStateRegistry。
基于此前 SavedStateRegistry < == > SavedStateProvider 建立的关系,由 SavedStateProvider 负责搜集 SavedStateHandle 中的 Map 数据 交给Bundle。
总结:系统回收组件时,ViewModel 对象会重新创建,数据可恢复。
任重而道远