在Android中注入ViewModelFactory有什么意义-Dagger 2

Nad*_*ada 4 java android kotlin dagger-2

我最近开始从事Android开发,并且来自iOS背景,诸如依赖注入框架和ViewModelFactories之类的概念对我来说是新事物。在我看到的所有教程中,ViewModels始终扩展了所提供的ViewModel类android.arch.lifecycle.ViewModel。如果viewModel具有参数,则必须创建一个ViewModelFactory并由Dagger注入到活动中

@Provides
@ActivityScope
fun provideViewModelFactory(dependency: Dependency) : ViewModelProvider.Factory = CustomViewModelFactory(dependency)
Run Code Online (Sandbox Code Playgroud)

CustomViewModelFactory然后将负责创建ViewModels。一切都很好,但是,当我可以将viewModels直接注入这样的活动中时,我并没有真正理解viewModelFactory的要点:

@Module
class ViewModelModule(private val user: User) {
    @ActivityScope
    @Provides
    fun provideMainViewModel() = MainViewModel(user = user)
    fun provideOtherViewModel() = OtherViewModel()
}

@ActivityScope
@Subcomponent(modules = [ViewModelModule::class])
interface MainActivitySubComponent {
    fun inject(activity: MainActivity)
}

@ApplicationScope
@Component()
interface ApplicationComponent {
    fun addMainActivitySubComponent(viewModelModule: ViewModelModule) : MainActivitySubComponent
}
Run Code Online (Sandbox Code Playgroud)

在我的活动中

class MainActivity : AppCompatActivity() {

    @Inject lateinit var viewModel: MainViewModel

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        val subComponent = (application as MainApplication).component.addMainActivitySubComponent(ViewModelModule(User("NEW NAME")))

        subComponent.inject(this)
    }
}
Run Code Online (Sandbox Code Playgroud)

那么,当我可以直接在活动中直接插入一个ViewModel时,ViewModelFactory有什么意义呢?

Dav*_*jak 5

让我们看一下注入ViewModel时会发生什么:

  1. 您的活动被创建(重新)。
  2. 您为所述活动创建一个(Sub)Component。
  3. 您注入依赖项。

有趣的部分是3,我们将注入一些作用域对象(例如Singletons),但为其余依赖项创建新对象。

每次(重新)创建活动(->配置更改)时,将再次创建这些对象,并且您将失去状态。您可以savedInstanceState用来保留一些数据,也可以找出其他方法来保存状态(例如,单例,保留的片段等)。


另一方面,ViewModel承诺在方向更改之间保持状态。当我们从ViewModelProviders方向更改后请求ViewModel时,我们将收到与以前相同的对象。我们不必担心保存和恢复状态。

我们将重新创建工厂,但是Android /支持库/ Jetpack / Arch组件(无论现在叫什么)都将存储Viewmodel并仅在之前未创建时创建它。先前的模型将在配置更改期间重用。


因此,如果您想直接注入ViewModel,显然可以这样做,但是ViewModel不会在方向更改之间保持其状态。


Epi*_*rce 5

您可以使用ViewModelProviders和 来ViewModelProviders.Factory确保在配置更改期间获得相同的 ViewModel 实例。因此,在 Activity 范围内,ViewModelProviders 创建的 ViewModel 仅创建一次。

它对于s 所具有的onCleared()回调的预期行为也是必需的ViewModel

在 Kotlin 中,与多重绑定相比,我更喜欢使用这种链接方法。尽管它确实要求您了解可以创建视图模型的“注入器”。