我试图通过ViewModel类型将KClass的子类绑定(bind)到 map 中:

@Module abstract class ViewModelModule {

    @Binds @IntoMap @ViewModelKey(MyViewModel::class)
    abstract fun bindsMyViewModel(viewModel: MyViewModel): ViewModel

    @Binds abstract fun bindViewModelFactory(factory: ViewModelFactory): ViewModelProvider.Factory

}

但是我收到了Dagger编译器错误:
e: ~/Example/app/build/tmp/kapt3/stubs/debug/com/example/app/injection/AppComponent.java:5: error: [dagger.android.AndroidInjector.inject(T)] java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
e:

e: public abstract interface AppComponent {
e:                 ^
e:       java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>> is injected at
e:           com.example.app.ui.ViewModelFactory.<init>(creators)
e:       com.example.app.ui.ViewModelFactory is injected at
e:           com.example.app.injection.ViewModelModule.bindViewModelFactory(p0)
e:       android.arch.lifecycle.ViewModelProvider.Factory is injected at
e:           com.example.app.ui.MyFragment.setViewModelFactory(p0)
e:       com.example.app.ui.MyFragment is injected at
e:           dagger.android.AndroidInjector.inject(arg0)

上面的ViewModelModule包含在我的AppModule中,它是AppComponent中的一个模块。因此Dagger应该能够提供Map<KClass<out ViewModel>, Provider<ViewModel>>所需的ViewModelFactory,但是我无法弄清楚它为什么崩溃。

我还尝试将ViewModelKey注释类切换到Java,将Class用作构造函数参数,而不是KClass。然后修改了我的ViewModelFactory以依赖于Map<Class<out ViewModel>, Provider<ViewModel>>,但是发生了相同的错误。

最佳答案

在批注中使用KClass时,实际上会将其编译为Java的Class。但是实际的问题是Kotlin编译器正在生成的java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>>中的通配符。

假设@ViewModelKey定义为

@MapKey
annotation class ViewModelKey(val value: KClass<out ViewModel>)

您需要将注入(inject)部位定义为
Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>

使用@JvmSuppressWildcards将阻止编译器生成通配符。

我实际上不知道,为什么Dagger编译器不支持通配符。您可以在此处看到类似的问题:Dagger 2: How to inject Map<Class<? extends Foo>, Provider<? extends Foo>>

09-30 15:03