Inject不适用于第二个构造函数

时间:2017-08-09 19:40:44

标签: kotlin dagger-2 dagger

对于我目前的项目,我正在使用Kotlin和Dagger 2。 我想在辅助构造函数中注入依赖项,但构造函数永远不会被初始化。

class SelectionFragmentModel ():ViewModel(){
   lateinit var channelInfosRepository: ChannelInfosRepository
   @Inject constructor(channelInfosRepository: ChannelInfosRepository) : this(){
      this.channelInfosRepository = channelInfosRepository
   }
   ...
}

作为一种解决方法,我目前正在主要构造函数中注入,但这不是最佳的。

class SelectionFragmentModel @Inject constructor(private val channelInfosRepository: ChannelInfosRepository):ViewModel(){
   constructor() : this(ChannelInfosRepository())
   ...
}

我错过了什么吗?

1 个答案:

答案 0 :(得分:0)

确保您的SelectionFragmentModel只有一个构造器。就Kotlin语言习语而言,构造函数是主要还是次要都没有关系。 SelectionFragmentModel中将只使用一个构造函数。

以下代码没有为初始化程序留下任何选项,因为只有一个使用哪个构造函数!

class SelectionFragmentModel: ViewModel {
    lateinit var channelInfosRepository: ChannelInfosRepository

    @Inject constructor(channelInfosRepository: ChannelInfosRepository) : super() {
        this.channelInfosRepository = channelInfosRepository
    }
}

示例(有效)

在此示例中,我们使用匕首进行了默认设置:

  1. @Module带注释的类,为我们提供ChannelInfosRepository;
  2. 修改后的SelectionFragmentModel(代码位于示例上方);
  3. @Component注释的接口,其中包含仅由一个模块组成的模块列表;
  4. 其他一切都是Android的东西。

这里是模块:

@Module
class AppModule {
    @Provides
    @Singleton
    fun providesChannelInfosRepository(): ChannelInfosRepository {
        return ChannelInfosRepository()
    }
}

@Component注释的接口:

@Singleton
@Component(modules = [AppModule::class])
interface AppComponent {
    fun inject(fragment: MainFragment)
}

这是实例化AppComponent的方式。 MyApplication中必须提到AndroidManifest.xml

class MyApplication : Application() {
    var appComponent: AppComponent? = null
        private set

    override fun onCreate() {
        super.onCreate()

        appComponent = DaggerAppComponent.builder()
                .appModule(AppModule())
                .build()
    }
}

使用SelectionFragmentModel注入AppComponent的片段:

class MainFragment : Fragment() {

    @Inject
    lateinit var selectionFragmentModel: SelectionFragmentModel

    companion object {
        fun newInstance() = MainFragment()
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // Ugly but fine for a test
        (activity?.application as? MyApplication)?.appComponent?.inject(this)
    }

    // onCreateView and other stuff ...

    @SuppressLint("SetTextI18n")
    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        println("ViewModel address = ${selectionFragmentModel}")
        println("ChannelInfosRepository address = ${selectionFragmentModel.channelInfosRepository}")
    }
}

结果(不是打印结果,而是在TextView中显示结果):

The result of the code in example