04 ViewModel组件架构原理解析

前言

ViewModel作为Jetpack组件库首屈一指的高频组件之一,我们有必要去了解他背后的工作原理,才能真正掌握它是如何实现存储数据的。它的出现释放了Activity/Fragment管理数据的压力,ViewModel经常会搭配LiveData一起用于MVVM的开发模式。

提纲
  • 我们是ViewModel;
  • ViewModel的用法;
  • ViewModel数据存储实现原理。
什么是ViewModel?

ViewModel具备宿主生命后期感知能力的数据存储组件,使用ViewModel保存的数据,在页面因配置变更导致页面销毁重建之后依然也是存在的。

Tips:配置变更主要是指:横竖屏切换、分辨率调整、权限变更、系统字体样式变更...

ViewModel的优势
页面配置更改数据不丢失

当设备因配置更改导致Activiy/Fragment重建,ViewModel中的数据并不会因此丢失,配合LiveData可以在页面重建后立马能收到最新保存的数据用以重新渲染页面。

生命周期感应

在ViewModel中难免会做一些网络请求或数据的处理,可以复写onCleared()方法,终止清理一些操作,释放内存,该方法在宿主onDestory时被调用。

数据共享

对于单Activity对Fragment的页面,可以使用ViewModel实现页面之间的数据共享,实际上不同的Activity也可以实现数据共享。


ViewModel的用法
使用ViewModel之前需要先添加依赖:
    //通常情况下,只需要添加appcompat就可以了
    api 'androidx.appcompat:appcompat:1.1.0'
    //如果想单独使用,可引入下面的依赖
    api 'androidx.lifecycler:lifecycle-viewmodel:2.0.0'
基本用法

存储的数据只能当页面因为配置变更导致的销毁再重建时刻复用,复用的是ViewModel的实例对象整体:

class HiViewModel() : ViewModel() {
    val liveData = MutableLiveData<List<GoodsModel>>()
    fun loadInitData():LiveData<List<GoodsModel>> {
        //from remote
        //为了适配因配置变更而导致的页面重建, 重复利用之前的数据,加快  新页面渲染,不再请求接口
        if(liveData.value==null){
           val remoteData = fetchDataFromRemote()
           liveData.postValue(remoteData)
        }
        return liveData
    }
  }

  //通过ViewModelProvider来获取viewmodel对象
  //如果在单Activity,多Fragment的页面,只需要都传递所在的Activity对象就可以获取到同一个ViewModel实例,从而实现数据共享。。
  val viewModel = ViewModelProvider(Activity/Fragment).get(HiViewModel::class.java)

  viewModel.loadPageData().observer(this,Observer{
       //渲染列表  
  })

ViewModel实现跨页面不同的(Activity)的数据共享

//让Application实现ViewModelStoreOwner 接口
class MyApp: Application(), ViewModelStoreOwner {
    private val appViewModelStore: ViewModelStore by lazy {
        ViewModelStore()
    }

    override fun getViewModelStore(): ViewModelStore {
        return appViewModelStore
    } 
}

val viewmodel = ViewProvider(application).get(HiViewModel::class.java)
ViewModel复用实现原理

上面说到,ViewModel可以实现因配置变更而导致页面销毁重建之后依然可以复用。准确点来说,应该是页面回复重建前后获取到的是同一个Viewmodel实例对象,以至于页面恢复重建后还能接着复用。那么这是为什么呢?

获取ViewModel实例的方式如下:

ViewModelProvider本质是从传递进去的ViewModelStore来获取实例。如果没有传递,则利用factory去创建一个新的,并存储到ViewModelStore。

val viewmodel =  ViewModelProvider(viewModelStore).get(HiViewModel::class.java)
//或者指定factory
val viewmodel =  ViewModelProvider(viewModelStore,factory).get(HiViewModel::class.java) 

不同的ViewModelFactory创建Viewmodel实例的方式不同:


ViewModelProvider获取ViewModel实例源码分析
class ViewModelProvider{
  private static final String DEFAULT_KEY =
          "androidx.lifecycle.ViewModelProvider.DefaultKey";

//根据传递的modelClass 构建一个默认的Key
public <T extends ViewModel> T get(Class<T> modelClass) {
        String canonicalName = modelClass.getCanonicalName();
        return get(DEFAULT_KEY + ":" + canonicalName, modelClass);
    }

//获取viewmodel实例时,也可以自行指定Key
public <T extends ViewModel> T get(String key, Class<T> modelClass) {
        ViewModel viewModel = mViewModelStore.get(key);
        if (mFactory instanceof KeyedFactory) {
            viewModel = ((KeyedFactory) (mFactory)).create(key, modelClass);
        } else {
            viewModel = (mFactory).create(modelClass);
        }
        mViewModelStore.put(key, viewModel);
        return (T) viewModel;
    }
}

ViewModelStore 一个真正用来存储ViewModel实例的集合。本质上是HashMap<String,ViewModel>

getViewModelStore源码分析

关键点来,想要ViewModel实例对象不随着宿主重建而销毁,那就要保证ViewModelStore实例对象不随着宿主重建而销毁。

class ComponentActivity extends ViewModelStoreOwner{
    static final class NonConfigurationInstances {
        Object custom;
        ViewModelStore viewModelStore;
    }
    public ViewModelStore getViewModelStore() {
       if (mViewModelStore == null) {
       //从源码上可以看出,会首先从`NonConfigurationInstances`来获取`ViewModelStore`实例对象,
       //如果不为空那是不是就能做到复用了 ?
       //所以重点在于`ViewModelStore`何时被存储到`NonConfigurationInstances`里面的.
           NonConfigurationInstances nc =
                   (NonConfigurationInstances) getLastNonConfigurationInstance();
           if (nc != null) {
               mViewModelStore = nc.viewModelStore;
           }
           if (mViewModelStore == null) {
               mViewModelStore = new ViewModelStore();
           }
       }
     
       return mViewModelStore;
    }
   }
}
onRetainNonConfigurationInstance源码分析

因系统原因页面被回收时,会触发该方法,所以ViewModelStore对象此时会被存储在NonConfigurationInstance中。在页面恢复重建时,会再次把这个NonConfigurationInstance 对象传递到新的Activity中实现对象复用。

class ComponentActivity{
public final Object onRetainNonConfigurationInstance() {
      Object custom = onRetainCustomNonConfigurationInstance();
      ViewModelStore viewModelStore = mViewModelStore;
      if (viewModelStore == null) {
          // 如果NonConfigurationInstance保存了viewModelStore,把它取出来
          NonConfigurationInstances nc =
                  (NonConfigurationInstances) getLastNonConfigurationInstance();
          if (nc != null) {
              viewModelStore = nc.viewModelStore;
          }
      }

      if (viewModelStore == null && custom == null) {
          return null;
      }

      NonConfigurationInstances nci = new NonConfigurationInstances();
      nci.custom = custom; 
        //把viewModelStore放到NonConfigurationInstances中并返回
      nci.viewModelStore = viewModelStore;
       //这样当页面被销毁时ViewModelStore就被保存起来了。
      return nci;
   }
}
总结

ViewModel和onSaveInstanceState方法有什么区别?

  • onSavedInstanceState只能存储轻量级的key-value键值对数据,非配置变更导致的页面被回收时才被触发,此时数据存储在ActivityRecord中;
  • ViewModel可以存放任意Object数据,因配置变更导致的页面被回收才有效。此时存在ActivityTheard#ActivityClientRecord中。

但是,如果是内存不足或者因为电量不足导致页面被回收,这种情况不是配置变更,所以ViewModel就无法实现复用了,那能不能让ViewModel在这种情况下也能实现数据的存储呢?必须可以,答案就在下一节!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352