写点什么

Jetpack MVVM 七宗罪 之三 :在 onViewCreated 中请求数据

用户头像
Android架构
关注
发布于: 19 小时前

}


}


如上,如果 ViewModel 在 onViewCreated 中请求数据,当 View 因为横竖屏等原因重建时会再次请求,而我们知道 ViewModel 的生命周期长于 View,数据可以跨越 View 的生命周期存在,所以没有必要随着 View 的重建反复请求。



[](


)正确的加载时机


===================================================================


ViewModel 的初次数据加载推荐放到 init{} 中进行,这样可以保证 ViewModelScope 中只加载一次


//TasksViewModel.kt


class TasksViewModel: ViewModel() {


private val _tasks = MutableLiveData<List<Task>>()


val tasks: LiveData<List<Task>> = _uiState


init {


viewModelScope.launch {


_tasks.value = withContext(Dispatchers.IO){


TasksRepository.fetchTasks()


}


}


}


}


[](


)LiveData KTX Builder


================================================================================


此外 lifecycle-livedata-ktx 提供的 LiveData KTX Builder 可以在创建 LiveData 的同时进行数据请求,无需创建 MutableLiveData,写法更简洁:


implementation “androidx.lifecycle:lifecycle-livedata-ktx:$latest_version”


val tasks: LiveData<Result> = liveData {


emit(Result.loading())


try {


emit(Result.success(repo.fetchData()))


} catch(ioException: Exception) {


emit(Result.error(ioException))


}


}


Note: 此种 KTX Builder 只适用于数据仅加载一次的情况,如果后续有用户动态触发的数据请求,则还需要借助 MutableLiveData 来实现。


[](


)设置 ViewModel 的初始化参数


===============================================================================


如果在 ViewModel 构造函数中请求数据,当需要参数时该如何传入呢?比如我们最开头例子中需要传入一个 TaskId。


[](


)1. 构造参数


====================================================================


最容易想到的方法是通过构造参数传入。


class DetailTaskViewModel(private val taskId: Int) : ViewModel() {


//...


init {


viewModelScope.launch {


_tasks.value = TasksRepository.fetchTask(taskId)


}


}


}


需要注意不能直接调用 ViewModel 的构造函数构造,这样无法将 ViewModel 存入 ViewModelStore。


此时需要定义一个 ViewModelProvider.Factory:


class TaskViewModelFactory(val taskId: Int) : ViewModelProvider.Factory {


override fun <T : ViewModel?> create(modelClass: Class<T>): T =


modelClass.getConstructor(Int::class.java)


.newInstance(taskId)


}


然后在 Fragment 中,用此 Factory 创建 ViewModel


class DetailTaskFragment : Fragment(R.layout.fragment_detailed_task){


《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
浏览器打开:qq.cn.hn/FTe 免费领取
复制代码


private val viewModel : DetailTaskViewModel by viewModels {


TaskViewModelFactory(requireArguments().getInt(TASK_ID))


}


override fun onViewCreated(view: View, savedInstanceState: Bundle?) {


super.onViewCreated(view, savedInstanceState)


//...


}


}


[](


)2. 使用 SavedStateHandler


====================================================================================


Fragment 1.2.0 或者 Activity 1.1.0 起, 可以使用 SavedStateHandle 作为 ViewModel 的参数。SavedStateHandle 可以帮助 ViewModel 实现数据持久化,同时可以传递 Fragment 的 arguments 给 ViewModel。


关于如何使用 SavedStateHandle 对数据进行持久化,由于不是本文重点不做介绍,这里只展示如何通过 SavedStateHandle 获取 arguments


implementation “androidx.lifecycle:lifecycle-viewmodel-savestate:$latest_version”


SavedStateHandle 版本的 ViewModel 定义如下:


class TaskViewModel(savedStateHandle: SavedStateHandle) : ViewModel() {


//...


init {


viewModelScope.launch {


_tasks.value = TasksRepository.fetchTask(


savedStateHandle.get<Int>(TASK_ID)


)


}


}


}


Fragment 中创建 ViewModel 如下:


class DetailTaskFragment : Fragment(R.layout.fragment_detailed_task){


private val viewModel: TaskViewModel by viewModels {


SavedStateViewModelFactory(


requireActivity().application,


requireActivity(),


arguments// 将 arguments 作为默认参数传递给 SavedStateHandler


)


}


override fun onViewCreated(view: View, savedInstanceState: Bundle?) {


super.onViewCreated(view, savedInstanceState)


//...


}


}


其中,SavedStateViewModelFactory 是关键,它会在构造 ViewModel 的时候,传入 SavedStateHandler


[](


)3. 自定义扩展方法


=======================================================================


前两种方法的模板代码较多,这里推荐一个自定义的扩展方法 viewModelByFactory,可以进一步简化代码


typealias CreateViewModel = (handle: SavedStateHandle) -> ViewModel


inline fun <reified VM : ViewModel> Fragment.viewModelByFactory(


defaultArgs: Bundle? = null,


noinline create: CreateViewModel = {


val constructor = findMatchingConstructor(VM::class.java, arrayOf(SavedStateHandle::class.java))


constructor!!.newInstance(it)


}


): Lazy<VM> {


return viewModels {


createViewModelFactoryFactory(this, defaultArgs, create)


}


}


inline fun <reified VM : ViewModel> Fragment.activityViewModelByFactory(


defaultArgs: Bundle? = null,


noinline create: CreateViewModel


): Lazy<VM> {


return activityViewModels {


createViewModelFactoryFactory(this, defaultArgs, create)


}


}


fun createViewModelFactoryFactory(


owner: SavedStateRegistryOwner,


defaultArgs: Bundle?,


create: CreateViewModel


): ViewModelProvider.Factory {


return object : AbstractSavedStateViewModelFactory(owner, defaultArgs) {


override fun <T : ViewModel?> create(key: String, modelClass: Class<T>, handle: SavedStateHandle): T {

用户头像

Android架构

关注

还未添加个人签名 2021.10.31 加入

还未添加个人简介

评论

发布
暂无评论
Jetpack MVVM七宗罪 之三 :在 onViewCreated 中请求数据