Android开发Jetpack组件ViewModel与LiveData使用讲解
目录
- 一、ViewModel
- 1、解决的问题
- 2、注意点事项
- 3、ViewModel案例
- 二、LiveData
- 1、viewmodel+livedata使用案例
- 2、viewmodel+livedata
一、ViewModel ViewModel是介于View(视图)和Model(数据模型)之间的中间层,能够使视图和数据分离,又能提供视图和数据之间的通信。如图所示:
文章图片
1、解决的问题
- 屏幕翻转后页面数据的丢失;
- 异步调用导致的内存泄露;
- 类膨胀提高维护难度和测试难度;
2、注意点事项
- ViewModel的生命周期要比Activity的生命周期更长,因此在使用ViewModel时,不要向ViewModel传入Activity的Context,因为这样会导致内存泄露。
- 如果必须使用Context,可使用AndroidViewModel中的Application。
3、ViewModel案例
【Android开发Jetpack组件ViewModel与LiveData使用讲解】我们知道,Android横竖屏切换时,当前的Activity会被销毁重建,然后Activity上面的数据将会全部丢失。(如Listview上面每个item的checkbox,横竖屏切换时,复选框就丢失所有选中信息)。一般,我们解决办法是,在配置清单Androidmanifest.xml的activity标签中加入android:configChanges="orientation|keyboardHidden",或在Activity里面通过复写onConfigurationChanged方法,实现在不同的屏幕状态下的处理方式。
那么,使用ViewModel将会是怎样的?
首先,创建一个连接层类MyViewModel.kt 继承AndroidViewModel,里面定义了number变量。
class MyViewModel(application: Application) : AndroidViewModel(application) {var number = 0 //}
然后,在activity中使用:
class TestActivity : AppCompatActivity() {private var textView: TextView? = nullprivate var viewModel: MyViewModel? = nulloverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)textView = findViewById(R.id.textView)//实例化viewModelviewModel =ViewModelProvider(this, AndroidViewModelFactory(application))[MyViewModel::class.java]//显示数据textView!!.text = String.valueOf(viewModel!!.number)}//点击事件fun plusNumber(view: View) {textView!!.text = String.valueOf(++viewModel!!.number)}}
实例化对应的viewModel后,通过访问viewModel里的属性number,不管怎么切换横竖屏,对应的屏幕上的数字都不会被清空。
二、LiveData 如上案例中,如果点击事件只是改变了viewmodel中的number值,text就不会刷新显示。LiveData就是一个能够在ViewModel中数据发生变化时通知页面刷新UI线程的组件库。LiveData和ViewModel的关系,如下图所示:
文章图片
1、viewmodel+livedata使用案例
首先,修改一下MyViewModel.kt
class MyViewModel : ViewModel() { //定义LiveData集合private var linkNumber: MutableLiveData? = null //得到LiveData集合fun getLinkNumber(): MutableLiveData ? {if (linkNumber == null) {//初始化linkNumber = MutableLiveData()linkNumber!!.value = https://www.it610.com/article/0}return linkNumber} //给外部提供修改集合内部属性的方法fun addLinkedNumber(n: Int) {linkNumber!!.value = linkNumber!!.value!! + n}}
在activity中使用:
class MainActivity : AppCompatActivity() {private var viewModel: MyViewModel? = nulloverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)val textView: TextView = findViewById(R.id.textView)viewModel =ViewModelProvider(this, AndroidViewModelFactory(application))[MyViewModel::class.java]//观察viewModel!!.getLinkNumber()!!.observe(this, Observer {textView.text = String.valueOf(it)})}fun reduce(view: View) {viewModel!!.addLinkedNumber(-1)}fun add(view: View) {viewModel!!.addLinkedNumber(1)}}
上面案例中,ViewModel的初始化不变。在点击事件中调用addLinkedNumber()方法,对ViewModel对应LiveData集合进行修改操作。viewModel!!.getLinkNumber()!!.observe(this, Observer {xx},来接受ViewModel发过来的修改通知,并及时更新至textView上。
2、viewmodel+livedata
优势总结
- 页面翻转数据状态保留
- 不再需要手动处理生命周期
- 数据始终保持最新状态
- 不会发生内存泄露
- 异步情况,不会因Activity停止而导致崩溃
推荐阅读
- SpringCloud微服务开发基于RocketMQ实现分布式事务管理详解
- iOS开发frame和bounds使用示例详解
- Android实现可折叠式标题栏
- Android使用AlarmManager设置闹钟功能
- Android中PopupWindow弹出式窗口使用方法详解
- android|壁纸服务的启动过程
- 共码未来(2022 Google 开发者大会亮点回顾)
- 开源代码安全 | 保护软件开发生命周期,你需要知道这些正确方法
- 前端成功转型后端开发的机会在哪里;如何评价框架TailwindCSS;前端的未来是远程工作吗|极客观点
- Linux|Ubuntu20.4(安装OpenCV4,配置vscode+CMake作为基本开发环境)