android - 如何创建一个可以从 ViewModel 和 Activity 访问的函数?
问题描述
我正在构建一个具有 MainActivity 和几个片段的 android 应用程序。我为每个片段使用 ViewModel 和 ViewModelFactory。我需要创建一个将执行网络请求的函数,并希望在其中一个片段中使用它,以便向用户显示进度条。当用户单击右上角菜单中的按钮时,我还想从覆盖函数 onOptionsItemSelected 中的 MainActivity 访问此函数。从这个按钮调用时,一切都应该在后台,并且在任务完成之前不需要向用户显示任何内容。
我尝试了两种方法。
- 将函数放在 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 而不是创建一个?
- 所以我的第二次尝试是把这个函数放在一个单独的文件中。然后我可以从 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?
解决方案
我会遵循您的选项 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))
推荐阅读
- swift - 了解 Codable 协议
- node.js - 如何从服务器端检索所有 FCM 令牌以向用户订阅主题?
- node.js - Azure nodejs Api 总是返回您要查找的资源已被删除、名称已更改或暂时不可用
- linux - 如何隐藏来自 shell 脚本的电子邮件中发送的 URL 的查询参数
- python - 在scrapy中以异步方法发出内联请求时无法使用`headers`或`dont_filter = True`
- awk - 如何使用awk对csv文件中的记录进行排序
- android - 如何在 Chromebook 上的 Android 应用程序的窗口边框上隐藏后退按钮
- css - Sweetalert 2中的输入在材料UI模式反应js中不可输入
- django - 如何使用通用 ListCreateAPIView 在嵌套序列化程序中更新用户及其配置文件?
- javascript - 如何从 dxf 文件解析 nurbs 表面?或者您知道用于解析它的库(对于 js,如果存在或任何其他语言)?