Android JetPack 深入分析ViewModel源码

news2025/1/22 8:40:19

文章目录

    • 前言
    • 源码分析
      • ViewModel是如何创建的?
        • ViewModelProvider(this)做了什么?
          • 小结
        • get(MyViewModel::class.java)做了什么?
          • 小结
      • ViewModel是如何实现配置更改后数据恢复的?
          • 整体时序图
    • 结语

前言

本文主要分析ViewModel相关源码,相关使用不再赘述,可参考Android ViewModel使用;

ViewModel 概览

Google官方给的ViewModel定义如下:

ViewModel类旨在`以注重生命周期的方式存储和管理界面相关数据`。

ViewModel类`让数据可在发生屏幕旋转等配置更改后继续留存`

定义主要提到两个关键点:

  • 生命周期
    ViewModel生命周期
    上图是Google官网提供的ViewModel生命周期图示,可以看到ViewModel的生命周期是从onCreate创建到完成并销毁Finished,开发中经常结合LiveData一起使用;

  • 配置更改后数据留存
    我们知道当屏幕旋转的时候,Activity会销毁重建,如果我们想恢复之前的数据,之前一般是通过onSaveInstanceState()来实现,内部是通过Bundle来实现,此方法仅适合可以序列化再反序列化的少量数据,而不适合数量可能较大的数据,而ViewModel可以很简单的实现这个功能,并存储较大的数据;

源码分析

ViewModel是如何创建的?

 	val viewModel = ViewModelProvider(this).get(MyViewModel::class.java)

以上是ViewModel创建代码

ViewModelProvider(this)做了什么?

### ViewModelProvider.java
    public ViewModelProvider(@NonNull ViewModelStoreOwner owner) {
        this(owner.getViewModelStore(), owner instanceof HasDefaultViewModelProviderFactory
                ? ((HasDefaultViewModelProviderFactory) owner).getDefaultViewModelProviderFactory()
                : NewInstanceFactory.getInstance());
    }

ComponentActivity则实现了ViewModelStoreOwnerHasDefaultViewModelProviderFactory接口;
因此我们只需要关注ComponentActivity中对应的方法;

  1. getViewModelStore()
  2. getDefaultViewModelProviderFactory()

我们先分析getViewModelStore:

	### ComponentActivity.getViewModelStore
   public ViewModelStore getViewModelStore() {
		...
        ensureViewModelStore();
        return mViewModelStore;
    }

可以看到getViewModelStore()调用了ensureViewModelStore(),其对应代码如下:

### ComponentActivity.ensureViewModelStore
    void ensureViewModelStore() {
        if (mViewModelStore == null) {
            NonConfigurationInstances nc =
                    (NonConfigurationInstances) getLastNonConfigurationInstance();
            if (nc != null) {
                // Restore the ViewModelStore from NonConfigurationInstances
                mViewModelStore = nc.viewModelStore;
            }
            if (mViewModelStore == null) {
                mViewModelStore = new ViewModelStore();
            }
        }
    }
   static final class NonConfigurationInstances {
        Object custom;
        ViewModelStore viewModelStore;
    }

ensureViewModelStore方法中,会判断mViewModelStore是否为null,如果为null的话,会先尝试调用getLastNonConfigurationInstance()方法获取,如果获取不到,则直接new创建一个ViewModelStore

我们看下ViewModelStore类:

public class ViewModelStore {

    private final HashMap<String, ViewModel> mMap = new HashMap<>();

    final void put(String key, ViewModel viewModel) {
        ViewModel oldViewModel = mMap.put(key, viewModel);
        if (oldViewModel != null) {
            oldViewModel.onCleared();
        }
    }

    final ViewModel get(String key) {
        return mMap.get(key);
    }

    Set<String> keys() {
        return new HashSet<>(mMap.keySet());
    }

    /**
     *  Clears internal storage and notifies ViewModels that they are no longer used.
     */
    public final void clear() {
        for (ViewModel vm : mMap.values()) {
            vm.clear();
        }
        mMap.clear();
    }
}

其内部创建了一个HashMap,以ViewModel的CanonicalName为key,ViewModel对象为value,存储Activity中创建的ViewModel对象;

我们再看下getDefaultViewModelProviderFactory()方法:

### ComponentActivity.getDefaultViewModelProviderFactory()
    public ViewModelProvider.Factory getDefaultViewModelProviderFactory() {
		...
        if (mDefaultFactory == null) {
            mDefaultFactory = new SavedStateViewModelFactory(
                    getApplication(),
                    this,
                    getIntent() != null ? getIntent().getExtras() : null);
        }
        return mDefaultFactory;
    }

主要是创建一个SavedStateViewModelFactory对象返回,我们看下SavedStateViewModelFactory的构造方法

    public SavedStateViewModelFactory(@Nullable Application application,
            @NonNull SavedStateRegistryOwner owner,
            @Nullable Bundle defaultArgs) {
        mSavedStateRegistry = owner.getSavedStateRegistry();
        mLifecycle = owner.getLifecycle();
        mDefaultArgs = defaultArgs;
        mApplication = application;
        mFactory = application != null
                ? ViewModelProvider.AndroidViewModelFactory.getInstance(application)
                : ViewModelProvider.NewInstanceFactory.getInstance();
    }

我们看下mFactory的创建,application 肯定不为null,因此会调用ViewModelProvider.AndroidViewModelFactory.getInstance(application)返回一个AndroidViewModelFactory对象

    public static class AndroidViewModelFactory extends ViewModelProvider.NewInstanceFactory {

        private static AndroidViewModelFactory sInstance;


        public static AndroidViewModelFactory getInstance(@NonNull Application application) {
            if (sInstance == null) {
                sInstance = new AndroidViewModelFactory(application);
            }
            return sInstance;
        }

        private Application mApplication;

      
        public AndroidViewModelFactory(@NonNull Application application) {
            mApplication = application;
        }

        @NonNull
        @Override
        public <T extends ViewModel> T create(@NonNull Class<T> modelClass) {
            if (AndroidViewModel.class.isAssignableFrom(modelClass)) {
                try {
                    return modelClass.getConstructor(Application.class).newInstance(mApplication);
                } catch (NoSuchMethodException e) {
                    throw new RuntimeException("Cannot create an instance of " + modelClass, e);
                } catch (IllegalAccessException e) {
                    throw new RuntimeException("Cannot create an instance of " + modelClass, e);
                } catch (InstantiationException e) {
                    throw new RuntimeException("Cannot create an instance of " + modelClass, e);
                } catch (InvocationTargetException e) {
                    throw new RuntimeException("Cannot create an instance of " + modelClass, e);
                }
            }
            return super.create(modelClass);
        }
    }

可以看到AndroidViewModelFactory主要是用来生产ViewModel的,其create()方法:根据传入的ViewModel类型【继承自AndroidViewModel或ViewModel】,通过反射创建ViewModel对象;

小结

调用ViewModelProvider(this)方法主要做了以下两件事:

  1. 获取ViewModelStore对象,如果ViewModelStore为null,会先尝试调用getLastNonConfigurationInstance()方法获取,获取不到则直接通过new创建,内部通过HashMap实现ViewModel对象的存储;

  2. 创建AndroidViewModelFactory对象,其内部提供了create(),使用反射创建对应的ViewModel对象;

get(MyViewModel::class.java)做了什么?

	### ViewModelProvider.java
   public <T extends ViewModel> T get(@NonNull Class<T> modelClass) {
        String canonicalName = modelClass.getCanonicalName();
        if (canonicalName == null) {
            throw new IllegalArgumentException("Local and anonymous classes can not be ViewModels");
        }
        return get(DEFAULT_KEY + ":" + canonicalName, modelClass);
    }

    @NonNull
    @MainThread
    public <T extends ViewModel> T get(@NonNull String key, @NonNull Class<T> modelClass) {
        ViewModel viewModel = mViewModelStore.get(key);

        if (modelClass.isInstance(viewModel)) {
            if (mFactory instanceof OnRequeryFactory) {
                ((OnRequeryFactory) mFactory).onRequery(viewModel);
            }
            return (T) viewModel;
        } else {
            if (viewModel != null) {
            }
        }
        if (mFactory instanceof KeyedFactory) {
            viewModel = ((KeyedFactory) mFactory).create(key, modelClass);
        } else {
            viewModel = mFactory.create(modelClass);
        }
        mViewModelStore.put(key, viewModel);
        return (T) viewModel;
    }
小结

get(MyViewModel::class.java)方法会根据传入的ViewModel的CanonicalName,先从mViewModelStore缓存中查找,如果找到,则直接返回,如果没找到,则调用前面创建的mFactory.create()方法通过反射创建对应的ViewModel,并添加到 mViewModelStore缓存中;

以上便是ViewModel相关的创建逻辑源码分析!

ViewModel是如何实现配置更改后数据恢复的?

我们知道当屏幕旋转配置发生变化时,会调用onDestroy(),在ComponentActivity构造方法中:

   getLifecycle().addObserver(new LifecycleEventObserver() {
            @Override
            public void onStateChanged(@NonNull LifecycleOwner source,
                    @NonNull Lifecycle.Event event) {
                if (event == Lifecycle.Event.ON_DESTROY) {
                    // Clear out the available context
                    mContextAwareHelper.clearAvailableContext();
                    // And clear the ViewModelStore
                    if (!isChangingConfigurations()) {
                        getViewModelStore().clear();
                    }
                }
            }
        });

当监听到Activity调用ON_DESTROY时,会调用getViewModelStore().clear(),清除ViewModelStore内部缓存的ViewModel对象,那为何重新创建后,ViewModel中持有的数据又没有丢失呢?

Activity销毁时会调用ActivityThread.handleDestroyActivity()方法:

  public void handleDestroyActivity(IBinder token, boolean finishing, int configChanges,
            boolean getNonConfigInstance, String reason) {
        ActivityClientRecord r = performDestroyActivity(token, finishing,
                configChanges, getNonConfigInstance, reason);
        if (r != null) {
         ...
        }
        mSomeActivitiesChanged = true;
    }

handleDestroyActivity方法中又会调用performDestroyActivity()方法:

  ActivityClientRecord performDestroyActivity(IBinder token, boolean finishing,
            int configChanges, boolean getNonConfigInstance, String reason) {
      		...
      		 ActivityClientRecord r = mActivities.get(token);
            if (getNonConfigInstance) {
                try {
                    r.lastNonConfigurationInstances
                            = r.activity.retainNonConfigurationInstances();
                } catch (Exception e) {
                    if (!mInstrumentation.onException(r.activity, e)) {
                        throw new RuntimeException(
                                "Unable to retain activity "
                                + r.intent.getComponent().toShortString()
                                + ": " + e.toString(), e);
                    }
                }
            }
     	...
        return r;
    }

performDestroyActivity方法中会调用activity.retainNonConfigurationInstances()

    NonConfigurationInstances retainNonConfigurationInstances() {
        Object activity = onRetainNonConfigurationInstance();
     
		...
        return nci;
    }

retainNonConfigurationInstances()方法中又调用onRetainNonConfigurationInstance(),其在ComponentActivity中具体实现代码如下:

    public final Object onRetainNonConfigurationInstance() {
        Object custom = onRetainCustomNonConfigurationInstance();
        ViewModelStore viewModelStore = mViewModelStore;
        if (viewModelStore == null) {
            NonConfigurationInstances nc =
                    (NonConfigurationInstances) getLastNonConfigurationInstance();
            if (nc != null) {
                viewModelStore = nc.viewModelStore;
            }
        }

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

        NonConfigurationInstances nci = new NonConfigurationInstances();
        nci.custom = custom;
        nci.viewModelStore = viewModelStore;
        return nci;
    }

可以看到会将viewModelStore保存到NonConfigurationInstances对象里返回,结合performDestroyActivity源码,最终会保存到ActivityClientRecord.lastNonConfigurationInstances属性中;

Activity重新创建的时候,会执行ActivityThread.performLaunchActivity()方法:

    private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
       			...
                activity.attach(appContext, this, getInstrumentation(), r.token,
                        r.ident, app, r.intent, r.activityInfo, title, r.parent,
                        r.embeddedID, r.lastNonConfigurationInstances, config,
                        r.referrer, r.voiceInteractor, window, r.configCallback,
                        r.assistToken);

 		...
        return activity;
    }

会调用 activity.attach()方法将lastNonConfigurationInstances对象传入,并赋值给Activity中的mLastNonConfigurationInstances属性;同时在ComponentActivity的构造方法中找到如下代码:

        getLifecycle().addObserver(new LifecycleEventObserver() {
            @Override
            public void onStateChanged(@NonNull LifecycleOwner source,
                    @NonNull Lifecycle.Event event) {
                ensureViewModelStore();
                getLifecycle().removeObserver(this);
            }
        });

ensureViewModelStore()方法上面我们已经看到,内部会先调用getLastNonConfigurationInstance()方法获取mViewModelStore对象,getLastNonConfigurationInstance()方法返回的正是mLastNonConfigurationInstances属性;从而实现ViewModel数据的恢复!

整体时序图

ViewModelStore时序图

结语

如果以上文章对您有一点点帮助,希望您不要吝啬的点个赞加个关注,您每一次小小的举动都是我坚持写作的不懈动力!ღ( ´・ᴗ・` )

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/699985.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何提高项目估算精准度 关键看3方面

项目估算非常重要&#xff0c;这直接关系着项目的成本和收入&#xff0c;如果估算不准确&#xff0c;将为项目带来较大风险。一般软件规模可以用多种方式进行估算&#xff0c;但是用功能点估算方式更准确&#xff0c;而自动估算让估算更快速&#xff0c;我们以CoCode需求分析工…

ChatGpt能够用来做什么

作为计算机从业人员&#xff0c;chatgpt能够从多方面提高大家的工作效率&#xff0c;主要包括以下几点&#xff1a; 技术问题解答&#xff1a;当遇到技术问题时&#xff0c;可以向ChatGPT提问并获取解答。ChatGPT可以提供相关的知识、文档和示例代码&#xff0c;帮助程序员快速…

vue2封装单张图片上传(常用于身份证正反面)

一.实现效果 二.入参 props: {defaultImg: {//默认位置的照片type: String,default: "",},uploadWidth: {//照片框的宽度type: String,default: "148px",},}, 另外如果修改了宽度的话&#xff0c;在外部组件需要用scss重写一下样式 /deep/ .el-upload-lis…

芯片中的上百亿个晶体管是如何设计的?

2021年4月21日&#xff0c;在芯片界的顶级会议Hot Chips大会上&#xff0c;Cerebras Systems公司发布了一款晶圆级引擎芯片——Wafer Scale Engine 2。 这款芯片采用台积电7纳米工艺制程&#xff0c;拥有85万个AI核心&#xff0c;包含2.6万亿个晶体管&#xff0c;面积为46225平…

Linux8.进程(中)(状态)

1.grep -v 关键字a :不显示关键字a匹配的信息。 2.进程状态 :新建&#xff0c;就绪&#xff0c;阻塞&#xff0c;挂起&#xff0c;执行&#xff0c;终止。 运行 : task_struct结构体在运行队列中排队&#xff0c;这就叫做运行态。 阻塞 :等待非CPU资源(磁盘&#xff0c;网卡…

科技云报道:公有云内卷时代,青云的新想象力在哪?

科技云报道原创。 2023年接踵而至的价格战&#xff0c;将国内公有云的竞争力度再次拉满。阿里云、腾讯云、京东云、移动云带头降价&#xff0c;也将寒意传导给了更多腰部云服务商。毫无疑问&#xff0c;这是一场对云服务商的残酷考验。 在公有云高度内卷的阶段&#xff0c;以…

kaggle新赛:肾脏血管分割大赛赛题解析(CV)

用AI为医疗贡献一份力量&#xff0c;从加入本次竞赛开始&#xff01; 赛题名称&#xff1a;HuBMAP - Hacking the Human Vasculature 从人肾组织切片中分割微血管结构 赛题链接&#xff1a;https://www.kaggle.com/competitions/hubmap-hacking-the-human-vasculature 赛题背…

java 正则表达式总结

目录 一、简介 二、源码分析 1.简单实例 : 2.底层实现 : 1 fund()方法 2 group(0/1)方法 (1)group(0): (2)group(1): 三、 基本语法 1.元字符介绍 : 2.元字符—字符匹配符 : Δ代码演示 3.关于字母大小写问题 : Δ代码演示 4.元字符—定位符 : 1 定义 2 常用定位符 3…

图简介-数据结构和算法教程

介绍 图是由顶点和边组成的非线性数据结构。顶点有时也被称为节点&#xff0c;并且边是连接图中的任何两个节点的线或弧。更正式地说&#xff0c;一个图是由一组顶点&#xff08;V&#xff09;和一组边&#xff08;E&#xff09;组成的。该图表示为G&#xff08;V&#xff0c;…

Win11的两个实用技巧系列之亮度条消的解决办法

Win11更新后无法调节亮度怎么办 Win11亮度条消的解决办法 Win11更新后无法调节亮度怎么办&#xff1f;win11系统升级以后&#xff0c;发现屏幕亮度不能调节&#xff0c;没有亮度调节按钮了&#xff0c;下面我们就来看看Win11亮度条消的解决办法 电脑更新后&#xff0c;亮度条消…

如何设计可以动态扩容缩容的分库分表方案?

对于分库分表来说&#xff0c;主要是面对以下问题&#xff1a; 选择一个数据库中间件&#xff0c;调研、学习、测试&#xff1b;设计你的分库分表的一个方案&#xff0c;你要分成多少个库&#xff0c;每个库分成多少个表&#xff0c;比如 3 个库&#xff0c; 每个库 4 个表&am…

SpringSecurity(五):前后端分离认证总结案例。

前后端分离认证总结案例 前言难点分析Controller层eneity层RoleUser dao层service层config层LoginFilterSecurityConfig resourcesmapper propertiespom.xml结尾 前言 和上一篇一样&#xff0c;从上倒下复制粘贴&#xff0c;所有代码贴完再运行&#xff0c;代码没有问题&#…

初步学习使用SpringBoot框架(手动插入数据模拟访问数据库)

对于SpringBoot框架介绍大家可以看看这个这篇文章&#xff0c;SpringBoot优缺点以及如何安装使用 以下我是按照老师给的安装方法进行安装使用SpringBoot框架&#xff1a; 大家安装SpringBoot框架时候&#xff0c;最好安装3.0以下的&#xff0c;不然需要对应较高版本的JDK版本&…

【Axure教程】拖动调整行高列宽的表格

表格是在系统软件中非常常用的工具。表格通常由行和列组成&#xff0c;用于以结构化的方式显示和组织数据。它们在各种场景中都有广泛的应用&#xff0c;包括数据分析、数据录入、报表生成、项目管理和数据可视化等领域。 今天作者就教大家如何在Axure里制作一个能通过鼠标拖动…

阿里巴巴最新 SpringCloudAlibaba 学习笔记,全程通俗易懂,一套搞懂!

Spring Cloud Alibaba 致力于提供微服务开发的一站式解决方案。此项目包含开发分布式应用微服务的必需组件&#xff0c;依托 Spring Cloud Alibaba&#xff0c;只需要添加一些注解和少量配置&#xff0c;就可以将 Spring Cloud 应用接入阿里微服务解决方案&#xff0c;通过阿里…

Visual C++中函数的覆盖和函数的隐藏

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天来聊聊Visual C中函数的覆盖和函数的隐藏。 Visual C中函数的覆盖和函数的隐藏与JAVA有很大不同&#xff0c;有些容易搞错的点和概念&#xff0c;我们要特别注意。 首先&#xff0c;先满足一下急性子的…

【刷题笔记——链表专栏】牛客网:合并两个排序链表

【刷题笔记——链表专栏】牛客网&#xff1a;合并两个排序链表 一、题目描述 二、思路分析 1、创建新的头节点newHead&#xff08;注意这一步&#xff09; 2、循环遍历pHead1和pHead2并进行链表并进行合并&#xff0c;直到pHead1和pHead2其中有一个为null时结束循环 3、合并完…

Win10,Win11玩游戏找不到d3dx9怎么解决

相信有些朋友遇到了d3dx9_42.dll丢失的情况不知道怎么解决&#xff0c;而今日小编带来的这篇文章就是讲解关于d3dx9_42.dll丢失进行修复的操作内容&#xff0c;d3dx9_42.dll丢失怎么解决&#xff1f;&#xff08;修复方法&#xff09;d3dx9 42.dll文件是DirectX中必备文件,许多…

【期末不挂科 学习数据结构】

期末不挂科 学习数据结构 第一章绪论1.1数据结构的基本概念1.1.1基本概念和术语1.数据2.数据元素3.数据对象4.数据类型5.数据结构 1.1.2数据结构三要素1.数据的逻辑结构2.数据的存储结构3.数据的运算 第一章绪论 1.1数据结构的基本概念 1.1.1基本概念和术语 1.数据 数据是信…

Redis7【⑧ Redis集群(cluster)】

Redis集群 Redis 集群是 Redis 数据库的分布式解决方案&#xff0c;它可以将数据分散存储在多个节点上&#xff0c;以支持大规模数据存储和高并发访问。 Redis 集群使用的是基于槽的分区策略&#xff0c;即将数据分成固定数量的槽&#xff0c;每个槽由一个主节点和多个从节点组…