ViewModel快速上手1-原生kotlin
ViewModel 原生支持 kotlin 案例
基本案例
viewmodel
是为了保存当当前 activity 切出或者销毁时,如何保存数据,以便下一次创建新的 activity 时进行调用
首先引入 lifecycle 依赖
implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'
之后,编写一个类继承 viewmodel,在这里配置数据的存取
我们需要实现:activity 调用该 vm,设置需要获取的水果序号,然后按照序号取出水果
代码清单 QuizViewModel.kt
package com.zhiyiyi.listviewdemo.modelimport android.util.Log
import androidx.lifecycle.ViewModelclass QuizViewModel : ViewModel() {// 初始化代码块,用于提示当前vm被调用init {Log.d("vm", "启动" + this.javaClass)}// 当当前的vm被销毁之前需要执行的方法override fun onCleared() {super.onCleared()}// 当前被选中的水果序号var currentIndex = 0// 简写get方法,获取指定序号的水果名称val getFruit: String get() = fruit[currentIndex]// 在这里存储水果数据private val fruit = listOf("apple", "pineapple", "pear", "orange", "banana")
}
viewmodel 全局仅存在一个实例,故配合懒加载,可以使用 val 定义 viewmodel 常量而非使用 var
故这里使用 by lazy
函数进行懒加载操作
因版本迭代,原 ViewModelProviders.of(this)
方法已被废弃,请使用以下代码中规定的新方法
代码清单 MainActivity.kt
package com.zhiyiyi.listviewdemoimport androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import androidx.lifecycle.ViewModelProvider
import com.zhiyiyi.listviewdemo.model.QuizViewModel
import kotlinx.android.synthetic.main.activity_main.*class MainActivity : AppCompatActivity() {private val quizVM: QuizViewModel by lazy {ViewModelProvider(this).get(QuizViewModel::class.java)}override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val quizVM = ViewModelProvider(this).get(QuizViewModel::class.java)btn_submit.setOnClickListener {val index = et_username.text.toString().toInt()quizVM.currentIndex = indexet_username.setText(quizVM.getFruit)}}
}
bundle 存储
当按下返回键并退出 activity 的时候,实际上该 activity 已经被销毁了; 然而直接点击主屏幕按键,只会挂起该 activity,并不会销毁它
oncreate
函数中的 savedInstanceState: Bundle?
参数还记得吗?
其中的 bundle 可看做是一个全局数据库,当 activity 被销毁后他会依然存在
利用此特性我们可以进行数据的恢复操作
因 activity 被销毁后,会调用 onSaveInstanceState 方法,而我们恰好可以为该方法传入值并存储他,当下一次 activity 被创建的时候,就可以直接取出我们存储的值
重写 onSaveInstanceState 方法,在 activity 被销毁前存储变量
override fun onSaveInstanceState(outState: Bundle) {super.onSaveInstanceState(outState)outState.putInt("index", quizVM.currentIndex)Log.d("vm", quizVM.currentIndex.toString())
}
在 oncreate 方法中直接从 bundle 中取出值
override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val quizVM = ViewModelProvider(this).get(QuizViewModel::class.java)// 从bundle中取出值val currentIndex = savedInstanceState?.getInt("index", 0) ?: 0quizVM.currentIndex = currentIndexet_username.setText(currentIndex.toString())...
}