使用匕首2.2实现MultiBinding时遇到问题。我在MVVM体系结构中使用了匕首。我注入了ViewModelProvideFactory
构造函数并绑定了模块的依赖关系。
我遵循了youtube上的Mitch教程
https://www.youtube.com/watch?v=DToD1W9WdsE&list=PLgCYzUzKIBE8AOAspC3DHoBNZIBHbIOsC&index=13
我已经在这些链接上搜索了解决方案,但是仍然面临着同样的问题。
Dagger2: ViewModel cannot be provided without an @Provides-annotated method
https://github.com/google/dagger/issues/1478
代码段
ViewModelKey
@MapKey
@Documented
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewModelKey {
Class<? extends ViewModel> value();
}
ViewModelFactoryModule
/**
* ViewModelFactoryModule responsible for providing [ViewModelProviderFactory]
*
* Annotated with Module to tell dagger it is a module to provide [ViewModelProviderFactory]
*
* Annotated with bind annotation to efficiently provide dependencies similar to provides annotation
*/
@Module
abstract class ViewModelFactoryModule {
@Binds
abstract fun bindViewModelFactory(viewModelFactory: ViewModelProviderFactory) : ViewModelProvider.Factory
}
ViewModelProviderFactory
@Singleton
class ViewModelProviderFactory @Inject
constructor(private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>) :
ViewModelProvider.Factory {
override fun <T : ViewModel> create(modelClass: Class<T>): T {
var creator: Provider<out ViewModel>? = creators[modelClass]
if (creator == null) { // if the viewmodel has not been created
// loop through the allowable keys (aka allowed classes with the @ViewModelKey)
for ((key, value) in creators) {
// if it's allowed, set the Provider<ViewModel>
if (modelClass.isAssignableFrom(key)) {
creator = value
break
}
}
}
// if this is not one of the allowed keys, throw exception
if (creator == null) {
throw IllegalArgumentException("unknown model class $modelClass")
}
// return the Provider
try {
return creator.get() as T
} catch (e: Exception) {
throw RuntimeException(e)
}
}
companion object {
private val TAG = "ViewModelProviderFactor"
}
}
StackTrace
> Task :app:kaptDebugKotlin FAILED
e: /Users/fazal/Documents/fazal/demo/AdvanceDagger2/app/build/tmp/kapt3/stubs/debug/com/example/advancedagger2/di/AppComponent.java:22: error: [Dagger/MissingBinding] java.util.Map<java.lang.Class<? extends androidx.lifecycle.ViewModel>,javax.inject.Provider<androidx.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
public abstract interface AppComponent extends dagger.android.AndroidInjector<com.example.advancedagger2.BaseApplication> {
^
java.util.Map<java.lang.Class<? extends androidx.lifecycle.ViewModel>,javax.inject.Provider<androidx.lifecycle.ViewModel>> is injected at
com.example.advancedagger2.viewmodel.ViewModelProviderFactory(viewModelsMap)
com.example.advancedagger2.viewmodel.ViewModelProviderFactory is injected at
com.example.advancedagger2.ui.AuthActivity.viewModelFactory
com.example.advancedagger2.ui.AuthActivity is injected at
dagger.android.AndroidInjector.inject(T) [com.example.advancedagger2.di.AppComponent → com.example.advancedagger2.di.ActivityBuilderModule_ContributeAuthActivity.AuthActivitySubcomponent]
我已经降级了Kotlin版本,但仍然面临相同的问题。告诉我,我做错了吗?
编辑1
我还通过AuthViewModel提供ViewModel,它属于活动范围。当活动破坏其组件,而依赖关系也破坏。
AuthViewModelModule
@Module
abstract class AuthViewModelModule {
/**
* Binds the auth view model dependency with [ViewModelKey] to group similar [ViewModel]
*
* Under the hood it is providing [com.example.advancedagger2.viewmodel.AuthViewModel]
*/
@Binds
@IntoMap
@ViewModelKey(AuthViewModel::class)
abstract fun bindAuthViewModel(authViewModel: AuthViewModel) : ViewModel
}
ActivityBuilderModule
/**
* This Class {@linkplain ActivityBuilderModule} is responsible for for android injection
* for the activity with in the application to avoid the seprate injection in each activity
*
* {@linkplain dagger.android.AndroidInjection#inject(Activity)}
*
* {@link com.example.advancedagger2.viewmodel.AuthViewModel} can be access from Auth Activity
* only so it is the concept of sub-modules
*
*/
@Module
public abstract class ActivityBuilderModule {
@ContributesAndroidInjector(
modules = AuthViewModelModule.class
)
abstract AuthActivity contributeAuthActivity();
}
答案 0 :(得分:2)
根据评论,我已经解决了问题
您的工厂可能不应该是@Singleton。它很轻巧 没有任何状态,并且您的视图模型可能绑定在 与活动相关的范围,反正在@Singleton中不可用
我用@Singleton
注释了工厂,该工厂在活动范围内不可用。我刚刚删除了@Singleton
注释。一切都按预期进行
答案 1 :(得分:0)
您需要为视图模型添加提供注释的方法,请参见下文。
@SuppressWarnings("unused")
@Binds
@IntoMap
@ViewModelKey(value = SomeViewModel.class)
abstract ViewModel bindSomeViewModel(SomeViewModel viewModel);
@SuppressWarnings("unused")
@Binds
abstract ViewModelProvider.Factory bindViewModelFactory(ViewModelFactory viewModelFactory);