使用 Hilt 进行依赖注入时如何将运行时参数传递给 ViewModel 的构造函数?
How to pass runtime parameters to a ViewModel's constructor when using Hilt for dependency injection?
我想知道如何在将 Hilt 用于 DI 时将运行时参数传递给 ViewModel 的构造函数?在使用 Hilt 之前,我有一个如下所示的 ViewModel:
class ItemViewModel(private val itemId: Long) : ViewModel() {
private val repo = ItemRepository(itemId)
}
class ItemViewModelFactory(private val itemId: Long) : ViewModelProvider.Factory {
@Suppress("unchecked_cast")
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
if (modelClass.isAssignableFrom(ItemViewModel::class.java)) {
return ItemViewModel(itemId) as T
}
throw IllegalArgumentException("Unknown ViewModel class")
}
我在我的片段中创建了上面的 ViewModel,如下所示:
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
val args: ItemScreenFragmentArgs by navArgs()
val itemId = args.itemId
//Create the view model factory
val viewModelFactory = ItemViewModelFactory(application, itemId)
// Get a reference to the ViewModel associated with this fragment.
val itemViewModel = ViewModelProvider(this, viewModelFactory).get(ItemViewModel::class.java)
}
如果我的 ItemViewModel 构造函数没有 itemId 参数,我使用 Hilt 的 ViewModel 和 Fragment 将如下所示:
class ItemViewModel
@ViewModelInject
constructor(private val repo: ItemRepository) : ViewModel() { }
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val itemViewModel: ItemViewModel by viewModels ()
}
我想知道如何将从 ItemFragment 的 NavArgs 获得的 itemId 传递给 ItemViewModel 的构造函数?有没有办法用 Hilt 做到这一点?
其实你可以使用工厂设计模式来创建需要item传递的对象
这是可行的,但我不确定它是否正确
class ItemRepository constructor(private val id: Int) {
}
class RepositoryFactory @Inject constructor() {
private var id: Int = 0
fun setId(id: Int) {
this.id = id
}
fun create(): ItemRepository = ItemRepository(id)
}
class ItemViewModel @ViewModelInject constructor(private val repositoryFactory: RepositoryFactory) : ViewModel() {
private var itemRepository: ItemRepository
init {
repositoryFactory.setId(45)
itemRepository = repositoryFactory.create()
}
}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val viewModel: ItemViewModel by viewModels ()
}
对于其他希望在使用 Dagger Hilt 时将运行时参数传递给 ViewModel 的人,我是这样做的:
我遵循了 this example which uses the AssistedInject 库中的代码。
我的代码现在如下所示:
class ItemViewModel
@AssistedInject
constructor(private val repo: ItemRepository, @Assisted private val itemId: Long) : ViewModel() {
init {
repo.itemId = itemId
}
@AssistedInject.Factory
interface AssistedFactory {
fun create(itemId: Long): ItemViewModel
}
companion object {
fun provideFactory(
assistedFactory: AssistedFactory,
itemId: Long
): ViewModelProvider.Factory = object : ViewModelProvider.Factory {
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
return assistedFactory.create(itemId) as T
}
}
}
}
@InstallIn(FragmentComponent::class)
@AssistedModule
@Module
interface AssistedInjectModule {}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val args: ItemScreenFragmentArgs by navArgs()
@Inject lateinit var itemViewModelAssistedFactory: ItemViewModel.AssistedFactory
private val itemViewModel: ItemViewModel by viewModels {
ItemViewModel.provideFactory(itemViewModelAssistedFactory, args.itemId)
}
}
Dagger 现在支持辅助注入,InflationInjection 也有自己的 repo。现在的语法类似于:
import dagger.assisted.Assisted
import dagger.assisted.AssistedFactory
import dagger.assisted.AssistedInject
...
class ItemViewModel
@AssistedInject
constructor(private val repo: ItemRepository, @Assisted private val itemId: Long) : ViewModel() {
init {
repo.itemId = itemId
}
//-@AssistedInject.Factory
@AssistedFactory
interface AssistedFactory {
fun create(@Named("item_id") itemId: Long): ItemViewModel
}
companion object {
fun provideFactory(
assistedFactory: AssistedFactory,
itemId: Long
): ViewModelProvider.Factory = object : ViewModelProvider.Factory {
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
return assistedFactory.create(itemId) as T
}
}
}
}
@InstallIn(FragmentComponent::class)
//-@AssistedModule
@Module
interface AssistedInjectModule {}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val args: ItemScreenFragmentArgs by navArgs()
@Inject lateinit var itemViewModelAssistedFactory:ItemViewModel.AssistedFactory
private val itemViewModel: ItemViewModel by viewModels {
ItemViewModel.provideFactory(itemViewModelAssistedFactory, args.itemId)
}
}
基于 Redek 的回答。更多相关信息 here
没有AssistedInject还有更好的解决方案,只需要使用SavedStateHandle。
如果您片段中的参数包含 userId:
val args: UserFragmentArgs by navArgs()
...
args.userId
然后只需在您的 savedStateHandle 中即可使用,无需额外工作。
class UserViewModel @Inject constructor (
private val repo: UserRepo,
private val state: SavedStateHandle
) : ViewModel() {
val user = repo.getUser(
state.get<String>("userId")
)
}
完整的实施细节:https://mattrobertson.dev/passing-safe-args-to-your-viewmodel-with-hilt-366762ff3f57
我想知道如何在将 Hilt 用于 DI 时将运行时参数传递给 ViewModel 的构造函数?在使用 Hilt 之前,我有一个如下所示的 ViewModel:
class ItemViewModel(private val itemId: Long) : ViewModel() {
private val repo = ItemRepository(itemId)
}
class ItemViewModelFactory(private val itemId: Long) : ViewModelProvider.Factory {
@Suppress("unchecked_cast")
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
if (modelClass.isAssignableFrom(ItemViewModel::class.java)) {
return ItemViewModel(itemId) as T
}
throw IllegalArgumentException("Unknown ViewModel class")
}
我在我的片段中创建了上面的 ViewModel,如下所示:
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
val args: ItemScreenFragmentArgs by navArgs()
val itemId = args.itemId
//Create the view model factory
val viewModelFactory = ItemViewModelFactory(application, itemId)
// Get a reference to the ViewModel associated with this fragment.
val itemViewModel = ViewModelProvider(this, viewModelFactory).get(ItemViewModel::class.java)
}
如果我的 ItemViewModel 构造函数没有 itemId 参数,我使用 Hilt 的 ViewModel 和 Fragment 将如下所示:
class ItemViewModel
@ViewModelInject
constructor(private val repo: ItemRepository) : ViewModel() { }
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val itemViewModel: ItemViewModel by viewModels ()
}
我想知道如何将从 ItemFragment 的 NavArgs 获得的 itemId 传递给 ItemViewModel 的构造函数?有没有办法用 Hilt 做到这一点?
其实你可以使用工厂设计模式来创建需要item传递的对象
这是可行的,但我不确定它是否正确
class ItemRepository constructor(private val id: Int) {
}
class RepositoryFactory @Inject constructor() {
private var id: Int = 0
fun setId(id: Int) {
this.id = id
}
fun create(): ItemRepository = ItemRepository(id)
}
class ItemViewModel @ViewModelInject constructor(private val repositoryFactory: RepositoryFactory) : ViewModel() {
private var itemRepository: ItemRepository
init {
repositoryFactory.setId(45)
itemRepository = repositoryFactory.create()
}
}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val viewModel: ItemViewModel by viewModels ()
}
对于其他希望在使用 Dagger Hilt 时将运行时参数传递给 ViewModel 的人,我是这样做的:
我遵循了 this example which uses the AssistedInject 库中的代码。
我的代码现在如下所示:
class ItemViewModel
@AssistedInject
constructor(private val repo: ItemRepository, @Assisted private val itemId: Long) : ViewModel() {
init {
repo.itemId = itemId
}
@AssistedInject.Factory
interface AssistedFactory {
fun create(itemId: Long): ItemViewModel
}
companion object {
fun provideFactory(
assistedFactory: AssistedFactory,
itemId: Long
): ViewModelProvider.Factory = object : ViewModelProvider.Factory {
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
return assistedFactory.create(itemId) as T
}
}
}
}
@InstallIn(FragmentComponent::class)
@AssistedModule
@Module
interface AssistedInjectModule {}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val args: ItemScreenFragmentArgs by navArgs()
@Inject lateinit var itemViewModelAssistedFactory: ItemViewModel.AssistedFactory
private val itemViewModel: ItemViewModel by viewModels {
ItemViewModel.provideFactory(itemViewModelAssistedFactory, args.itemId)
}
}
Dagger 现在支持辅助注入,InflationInjection 也有自己的 repo。现在的语法类似于:
import dagger.assisted.Assisted
import dagger.assisted.AssistedFactory
import dagger.assisted.AssistedInject
...
class ItemViewModel
@AssistedInject
constructor(private val repo: ItemRepository, @Assisted private val itemId: Long) : ViewModel() {
init {
repo.itemId = itemId
}
//-@AssistedInject.Factory
@AssistedFactory
interface AssistedFactory {
fun create(@Named("item_id") itemId: Long): ItemViewModel
}
companion object {
fun provideFactory(
assistedFactory: AssistedFactory,
itemId: Long
): ViewModelProvider.Factory = object : ViewModelProvider.Factory {
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
return assistedFactory.create(itemId) as T
}
}
}
}
@InstallIn(FragmentComponent::class)
//-@AssistedModule
@Module
interface AssistedInjectModule {}
@AndroidEntryPoint
class ItemFragment : Fragment() {
private val args: ItemScreenFragmentArgs by navArgs()
@Inject lateinit var itemViewModelAssistedFactory:ItemViewModel.AssistedFactory
private val itemViewModel: ItemViewModel by viewModels {
ItemViewModel.provideFactory(itemViewModelAssistedFactory, args.itemId)
}
}
基于 Redek 的回答。更多相关信息 here
没有AssistedInject还有更好的解决方案,只需要使用SavedStateHandle。
如果您片段中的参数包含 userId:
val args: UserFragmentArgs by navArgs()
...
args.userId
然后只需在您的 savedStateHandle 中即可使用,无需额外工作。
class UserViewModel @Inject constructor (
private val repo: UserRepo,
private val state: SavedStateHandle
) : ViewModel() {
val user = repo.getUser(
state.get<String>("userId")
)
}
完整的实施细节:https://mattrobertson.dev/passing-safe-args-to-your-viewmodel-with-hilt-366762ff3f57