首页 > 解决方案 > 如何创建一个可以从 ViewModel 和 Activity 访问的函数?

问题描述

我正在构建一个具有 MainActivity 和几个片段的 android 应用程序。我为每个片段使用 ViewModel 和 ViewModelFactory。我需要创建一个将执行网络请求的函数,并希望在其中一个片段中使用它,以便向用户显示进度条。当用户单击右上角菜单中的按钮时,我还想从覆盖函数 onOptionsItemSelected 中的 MainActivity 访问此函数。从这个按钮调用时,一切都应该在后台,并且在任务完成之前不需要向用户显示任何内容。

我尝试了两种方法。

  1. 将函数放在 ViewModel 中。这非常适合从片段调用并且一切正常。但是当从 Main 活动调用时,我似乎无法正确引用 ViewModel。

我在 onCreate 之前把它放在 MainActivity

private lateinit var syncViewModel: SyncViewModel

这在 onCreate

syncViewModel = ViewModelProvider(this).get(SyncViewModel::class.java)

我收到以下错误消息。

java.lang.RuntimeException:无法启动活动 ComponentInfo java.lang.RuntimeException:无法创建类 com 的实例。. ._.SyncViewModel

如何从活动中访问已经创建的 ViewModel 而不是创建一个?

  1. 所以我的第二次尝试是把这个函数放在一个单独的文件中。然后我可以从 ViewModel 和 MainActivity 访问它。但是我找不到更新 ViewModel 中进度条的方法。

在 ViewModel 中我有以下内容,然后这个片段中的 xml 引用了这个 syncProgress 值。

private var _syncProgress = MutableLiveData<Int>()

val syncProgress: LiveData<Int> = _syncProgress

我缺少什么来连接这个单独的函数以更新我的 ViewModel 中的值?

似乎选项#1是正确的方法,但也许有第三种我不知道的方法会更好!任何帮助表示赞赏!

更新 1

我的 ViewModel 定义为

class SyncViewModel(
val database: RoomDatabase,
application: Application) : AndroidViewModel(application) {

在片段中

// Create an Instance of the ViewModel Factory
val dataSource = RoomDatabase.getDatabase(application)
val viewModelFactory = SyncViewModelFactory(dataSource, application)

private lateinit var syncViewModel: SyncViewModel
syncViewModel = ViewModelProvider(this,viewModelFactory).get(SyncViewModel::class.java)

但我不确定我需要如何修改 MainActivity 中的 SyncViewModel?

标签: androidkotlinandroid-fragmentsmvvmviewmodel

解决方案


我会遵循您的选项 1,并使用视图模型提供程序委托来简化获取您的参考。您应该在 Fragment 中指定一个 Activity 范围的 ViewModel,以便它获得与 Activity 相同的实例。

在您的活动中:

private val syncViewModel: SyncViewModel by viewModels()

在你的片段中:

private val syncViewModel: SyncViewModel by activityViewModels()

由于可以使用 Application 实例获取数据库,因此可以通过将其移出构造函数来避免创建工厂,如下所示:

class SyncViewModel(application: Application) : AndroidViewModel(application) {

    val database = RoomDatabase.getDatabase(application)

如果您确实想继续使用您的工厂,可以将其作为参数传递给委托函数,例如:

private val dataSource = RoomDatabase.getDatabase(application)
private val syncViewModel: SyncViewModel by activityViewModels(SyncViewModelFactory(dataSource, application))

推荐阅读