1

我正在使用ViewModel更新操作栏中的标题

共享视图模型

class SharedViewModel @ViewModelInject constructor(
    @Assisted private val savedStateHandle: SavedStateHandle
) : ViewModel() {

    val title: MutableLiveData<String> by lazy {
        MutableLiveData<String>()
    }

    val backButton: MutableLiveData<Boolean> by lazy {
        MutableLiveData<Boolean>()
    }
}

MainActivity 观察者

@AndroidEntryPoint
...
sharedViewModel.title.observe(this, Observer {
    supportActionBar?.title = it
})

使用下面的代码似乎在Fragment(在调试器中检查)创建了一个新实例:

@AndroidEntryPoint
...
private val viewModel: SharedViewModel by viewModels()

但似乎以这种方式工作

val viewModel = ViewModelProvider(requireActivity()).get(SharedViewModel::class.java)

文章参考

这应该以这种方式完成还是我做错了什么?

谢谢!

4

1 回答 1

8

如果你想在片段中使用共享视图模型,你必须使用by activityViewModels()而不是by viewModels().

为什么下一行有效但by viewModels()无效?

ViewModelProvider(requireActivity()).get(SharedViewModel::class.java)

因为默认情况下viewModels()ownerProducer参数值为{ this }. 为了更好地理解它,这里是源代码:

@MainThread
inline fun <reified VM : ViewModel> Fragment.viewModels(
    noinline ownerProducer: () -> ViewModelStoreOwner = { this },
    noinline factoryProducer: (() -> Factory)? = null
) = createViewModelLazy(VM::class, { ownerProducer().viewModelStore }, factoryProducer)

如您所见,它是Fragment类的扩展功能。这意味着 ViewModelStoreOwner片段。一旦从堆栈中删除片段,它存储在视图模型存储中的所有视图模型都消失了。

但是,如果您使用by activityViewModels()Activity作为视图模型店主使用。注意默认情况下使用 ofrequireActivity().viewModelStore而不是ownerProducer().viewModelStorewhich Fragment

@MainThread
inline fun <reified VM : ViewModel> Fragment.activityViewModels(
    noinline factoryProducer: (() -> Factory)? = null
) = createViewModelLazy(VM::class, { requireActivity().viewModelStore },
    factoryProducer ?: { requireActivity().defaultViewModelProviderFactory })

最终,createViewModelLazy正在调用

ViewModelProvider(store, factory).get(viewModelClass.java)

这相当于您使用手动创建视图模型

ViewModelProvider(requireActivity()).get(SharedViewModel::class.java)

如何解决问题?

使用by activityViewModels()

@AndroidEntryPoint
...
private val viewModel: SharedViewModel by activityViewModels()

注意:by activityViewModels()并且by viewModels()只是ViewModelProvider(...).get(...).

于 2020-08-04T09:56:33.480 回答