Jetpack 系列——ViewModel
}
}
可以看到,ViewModel 类只有一个生命周期方法,那就是 onCleared(),我们通常需要在这个方法中进行一些资源的释放,避免内存泄漏。
要注意的是,Activity 的生命周期在变化的时候,并不会执行 onCleared()。为了证明 Activity 在执行各生命周期时,ViewModel 并不会随之变化,所以我们可
以在 ViewModel 中使用 Handler 或者 RxJava 做一个定时循环任务,观测 Activity 是否会影响 ViewModel:
class MyViewModel : ViewModel() {
var handler: Handler = object : Handler() {
var i = 0
override fun handleMessage(msg: Message) {
super.handleMessage(msg)
print(i++)
sendEmptyMessageDelayed(0, 500)
}
}
public fun startHandler() {
handler.sendEmptyMessageDelayed(0, 500)
}
override fun onCleared() {
super.onCleared()
print("onCleared")
handler.removeMessages(0)
}
}
在 MyViewModel 中使用 Handler 创建了一个定时轮训任务,每 500 毫秒打印一遍日志。接下来,在 Activity 中创建出 ViewModel 实例对象,调用其 startHandler()方法开始执行:
class ViewModelActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_view_model)
val viewModel = ViewModelProvider(this).get(MyViewModel::class.java)
viewModel.startHandler()
}
}
运行代码会发现,当 Activity 发生屏幕旋转时,日志并没有中断,且打印出的数字连续,所以证明 ViewModel 不受 Activity 的生命周期影响。
上述代码中也介绍了最重要的一个点,那就是 ViewModel 实例的创建,首先需要创建一个 ViewModelProvider 类对象,其构造方法中需要传入 Activity 实例,在 androidx 中,FragmentActivity 默认实现了 ViewModelStoreOwner 接口:
public class ComponentActivity extends androidx.core.app.ComponentActivity implements
LifecycleOwner,
ViewModelStoreOwner,
HasDefaultViewModelProviderFactory,
SavedStateRegistryOwner,
OnBackPressedDispatcherOwner {
}
创建 ViewModelProvider 实例后,调用其 get()方法就可以:
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();
}
}
get()方法其核心是在调用 ViewModelStore 的 get()方法,可以看出 ViewModelStore 类就是使用 HashMap 将 ViewModel 与其 key(class 名称)保存了起来。
评论