1.官方文档

  MVVM 官方文档:  https://developer.android.com/jetpack/docs/guide

  ViewModel 文档:  https://developer.android.com/topic/libraries/architecture/viewmodel

  ViewModel保存状态:    https://developer.android.com/topic/libraries/architecture/viewmodel-savedstate

  LiveData 官方文档: https://developer.android.com/topic/libraries/architecture/livedata 

2.android mvvm 软件架构简介

2.1 架构图示

  

  其中数字的简单描述如下:

    1. View通过ViewModel获取数据对象展示,并成为它的观察者。
    2. ViewModel通过Model得到真正的数据对象,根据业务需求通过model修改数据
    3. Model 只负责数据的操作:查找、从服务器下载新数据、修改等操作。

2.2 View

  • 负责数据显示、用户交互,系统交互。android系统中的Fragment 、Activity。
  • 只通过ViewModel得到目标数据对象展示并成为它的观察观察者。数据对象存在LiveData<XX>中。
  • 不处理业务逻辑,不打开数据库,不通过网络请求数据等.

2.3 Model

  • 数据抽象,封装
  • 数据存储(文件、内存、服务器、本地数据库等)、数据提取
  • 为ViewModel返回真正的数据对象(LiveData<xxx>)

2.4 ViewModel

  • 负责为View(View(Fragment 或 Activity))准备数据
  • 处理业务逻辑,不直接处理数据。
  • 与模型进行通信,后者存储、网络请求数据等

2.5 LiveData

  • 负责关联view与viewModel

3.LiveData

  • LiveData 一系列类模板,负责关联view与viewModel,采用观察者模式、类里存放被观察的数据,它的子类如下。

    

  • 具有系统组件(activity,fragment,service)生命周期感知能力(无需手动),系统确保只会将更新通知给活跃的观察者。
  • 不支持非主线程更新
 -- ::54.688 -/com.example.frgmtvm E/AndroidRuntime: FATAL EXCEPTION: Timer-
Process: com.example.frgmtvm, PID:
java.lang.IllegalStateException: Cannot invoke setValue on a background thread
at androidx.lifecycle.LiveData.assertMainThread(LiveData.java:)
at androidx.lifecycle.LiveData.setValue(LiveData.java:)
at androidx.lifecycle.MutableLiveData.setValue(MutableLiveData.java:)
at com.example.frgmtvm.ui.main.MainViewModel$test1$.run(MainViewModel.kt:)
at java.util.TimerThread.mainLoop(Timer.java:)
at java.util.TimerThread.run(Timer.java:)

4.ViewModel

  • 如果在Activity、Fragment中保存数据,容易引起3个问题:

    • 当它们被销毁后再次重新构建的时候,需要重新请求加载数据,造成资源的浪费,
    • 如果请求数据是异步的,在销毁Activity时,容易引起内存泄漏。
    • Activity、Fragmen代码膨胀、不易维护、测试。
  • android中 ViewModel 类旨在以注重生命周期的方式存储和管理界面相关的数据。让数据可在发生屏幕旋转等配置更改后继续留存在.
  • 绝不能引用视图、Lifecycle 或可能存储对 Activity 上下文的引用的任何类
  • ViewModel 对象应该比它们相应 View 对象存在的时间更长,因此 ViewModel 实现中不得包含对 View 对象的直接引用。
  • 如果 ViewModel 需要 Application 上下文(例如,为了查找系统服务),可以扩展 AndroidViewModel 类
  • ViewModel 对象存在的时间范围是获取 ViewModel 时传递给 ViewModelProvider 的 ViewModelStoreOwner。ViewModel 将一直留在内存中,直到限定其存在时间范围的 ViewModelStoreOwner 永久消失:对于 Activity,是在 Activity finish时;而对于 Fragment,是在 Fragment  detach时。
  • SavedStateHandle 用来保存轻量的页面状态(key-value),可以把它们传给VIewModel 使用。

5.示例代码

完整代码:

  https://github.com/f9q/mvvm

5.1 第1步:定义Model

  • User

     class User {
    var name = "TEST-"
    var age = override fun toString(): String {
    return "$name-$age"
    }
    }
  • UserModel, 数据对象用LiveData系列类保存

     class UserModel {
    
         var userData    =   MediatorLiveData<User>()
    
         init {
    val user = User()
    userData.value = user
    } fun userFromSql() : LiveData<User> {
    return userData
    }
    fun userFromServer() : LiveData<User>{
    return userData
    }
    fun userFromFile() : LiveData<User>{
    return userData
    }
    fun userFromCache() : LiveData<User>{
    return userData
    } fun modifyUser(age : Int,name : String){
    val user = userData.value
    user?.age = age
    user?.name = name userData.value = user
    } }

5.2 第2步:定义ViewModel  

定义ViewModel,保存model对象,并提供返回数据对象的方法

 class MainViewModel : ViewModel{

     var userModel   :   UserModel  = UserModel()

     fun loadUser() : LiveData<User>{
val ud = userModel.userFromServer()
return ud
}
override fun onCleared() {
super.onCleared()
Log.e("MainViewModel","onCleared")
cancelTask()
}
//...
}

5.3 第3步:关联View与ViewModel

在android中,Fragment、activity被当作View

 class Fragment1 : Fragment() {

     lateinit var nameKey    : TextView
lateinit var nameValue : EditText
lateinit var ageKey : TextView
lateinit var ageValue : TextView val viewModel : MainViewModel by lazy { initViewModel() } fun initViewModel() : MainViewModel{
// return ViewModelProvider(this).get(MainViewModel::class.java)
return ViewModelProvider(activity!!).get(MainViewModel::class.java)
} fun bindViewModel(){
viewModel.loadUser().observe(viewLifecycleOwner){user->
nameValue.setText(user.name)
ageValue.setText(user.age.toString())
}
}
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View {
val v = inflater.inflate(R.layout.view1, container, false)
initView(v)
return v
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
bindViewModel()
}
//...
}

  其中:

  • 第11行,绑定view与viewModel时使用的ViewModelStoreOwner是Fragment,viewModel的生命期与Fragment关联,退出Fragment后,数据失效
  • 第12行,使用ViewModelStoreOwner 是 Activity,该viewModel的生命期长,退出Activity才失效,可用于多个fragment 共享数据。
  • 第16行,关联view与对应的数据,当数据变化时,这里会得到通知,然后刷新界面。

  

android MVVM(1)用LiveData关联VM 与 V的更多相关文章

  1. android MVVM(2)用数据绑定关联VM 与 V

    1.官方文档 https://developer.android.com/topic/libraries/data-binding/architecture 2.简介 数据绑定库 可与MVVM 架构组 ...

  2. Android MVVM框架RoboBinding初探

    RoboBinding是一个实现了数据绑定 Presentation Model(MVVM) 模式的Android开源框架.MVVM模式是MVC模式的重要更新,使得项目结构更加的优美,易于维护以及方便 ...

  3. 如何构建Android MVVM 应用框架

    概述 说到Android MVVM,相信大家都会想到Google 2015年推出的DataBinding框架.然而两者的概念是不一样的,不能混为一谈.MVVM是一种架构模式,而DataBinding是 ...

  4. (Android MVVM)使用Data Binding Library(2)

    复习 上一篇学到了如何在layout.xml文件中增加元素,实现数据绑定,本篇接着学习. 事件处理 在layout.xml上绑定事件有两种方法,各有千秋. 1.方法引用 2.监听绑定 1.使用方法引用 ...

  5. Android Jetpack组件 - ViewModel,LiveData使用以及原理

    本文涉及的源码版本如下: com.android.support:appcompat-v7:27.1.1 android.arch.lifecycle:extensions:1.1.1 android ...

  6. Android开发利器之Data Binding Compiler V2 —— 搭建Android MVVM完全体的基础

    原创声明: 该文章为原创文章,未经博主同意严禁转载. 前言: Android常用的架构有:MVC.MVP.MVVM,而MVVM是唯一一个官方提供支持组件的架构,我们可以通过Android lifecy ...

  7. android mvvm初探

    目前google的databinding library还处在rc版,其中编译器发挥了主要作用.目前也只是在android studio开发环境中支持. mvvm能够大大降低模块间的耦合度,在开发过程 ...

  8. Android MVVM小结

    一.概念 关于MVC.MVP与MVVM的概念就不介绍了,总之一句话,MVVM概念出现比MVP早,MVP比MVC早,作为程序员就应该去学习最新的技术不是?详细的概念介绍移步这里吧,https://www ...

  9. Android ToolBar自定义图标,关联DrawerLayout

    Android5.0出现了一个可以代替ActionBar的控件ToolBar,使用更加灵活,一般我们使用ToolBar来和DrawerLayout配合使用,官方提供了一个开关类ActionBarDra ...

随机推荐

  1. 自动发布-asp.net自动发布、IIS站点自动发布(集成SLB、配置管理、Jenkins)

    PS:概要.背景.结语都是日常“装X”,可以跳过直接看自动发布 环境:阿里云SLB.阿里云ECS.IIS7.0.Jenkins.Spring.Net 概要 公司一个项目从无到有,不仅仅是系统从无到有的 ...

  2. lombok的基本使用方法

    在java刚开始学习的时候,首先就会学习封装.继承和多态,就拿封装来说,封装就是为了保护数据安全而将实体类内部数据保持为私有状态,如果外部程序想要访问里面的数据就必须调用此实体类提供的相关数据接口,这 ...

  3. Java基础—字符串

    事实上,Java是没有内置的字符串类型的,而是在标准Java类库中提供了一个预定义类String.每个用双引号括起来的字符串都是String类的一个实例: String str = "&qu ...

  4. Vue中diff算法的理解

    Vue中diff算法的理解 diff算法用来计算出Virtual DOM中改变的部分,然后针对该部分进行DOM操作,而不用重新渲染整个页面,渲染整个DOM结构的过程中开销是很大的,需要浏览器对DOM结 ...

  5. 精讲RestTemplate第4篇-DELETE、PUT等请求方法使用详解

    本文是精讲RestTemplate第5篇,前篇的blog访问地址如下: 精讲RestTemplate第1篇-在Spring或非Spring环境下如何使用 精讲RestTemplate第2篇-多种底层H ...

  6. 密码学系列——消息摘要(c#代码实操)

    前言 简介: 消息摘要(Message Digest)又称为数字摘要(Digital Digest) 它是一个唯一对应一个消息或文本的固定长度的值,它由一个单向Hash加密函数对消息进行作用而产生 使 ...

  7. DB2 创建存储过程保存:XX 后面找到异常标记 "END-OF-STATEMENT"。

    存储过程最后一行加结束符@: 然后执行:db2 -td@ -vf /home/WGJ/proc_data_calculate.sql [slsadmin@localhost /]$ db2 -td@ ...

  8. C# ASP 异步存储数据

    1.异步委托 在导航栏接收到提交的请求后,调用个各子画面的保存答案方法,之后实例化委托 saveToDB . 当执行BeginInvoke后,服务器会另起线程执行saveToDB里的的方法,因为这里要 ...

  9. Vue 函数式组件 functional

    函数式组件 无状态 无法实例化 内部没有任何生命周期处理函数 轻量,渲染性能高,适合只依赖于外部数据传递而变化的组件(展示组件,无逻辑和状态修改) 在template标签里标明functional 只 ...

  10. 存储系列之 XFS文件系统简介

    引言:磁盘容量越来越大,文件系统管理的文件也是越来越大.越来越多,如何破解?唯有快!于是动态分配.B+树开始登上舞台.还记得当年MySQL的索引结构吗,好的作品所见略同. 一.XFS为什么替换Ext4 ...