使用KClass类型绑定到映射

时间:2017-06-19 19:56:30

标签: android dependency-injection kotlin dagger-2 dagger

我正在尝试将ViewModel的子类按其KClass类型绑定到地图中:

@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是我Map<KClass<out ViewModel>, Provider<ViewModel>>中的一个模块。所以Dagger应该能够提供ViewModelFactory所需的ViewModelKey,但我无法弄清楚它为什么会崩溃。

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

1 个答案:

答案 0 :(得分:38)

在注释中使用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>)

您需要将注射部位定义为

Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>

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

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