LiveData,ViewModel,Lifecycle原理解析

作者: CharlesYale | 来源:发表于2019-01-19 19:57 被阅读13次

    一. 使用

    Lifecycle如果用Activity/Fragment的话,可以不必理会,直接用support包版本26.1.0以上的AppCompactActivity/FragmentActivity(继承自SupportActivity)或者Fragment即可

    一般由Activity在onCreate()的时候通过ViewProviders.of(this).get(xxxViewModel.class)创建,例如

    public class MyActivity extends FragmentActivity {
     
        MyActivity mViewModel;
     
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_my);
     
            mViewModel = ViewModelProviders.of(this).get(MyActivityViewModel.class);
        }
    }
    

    Activity直接调用ViewModel去做一些业务逻辑,并通过监听ViewModel中的LiveData变化来更新界面

    public class MyActivity extends FragmentActivity {
     
        MyActivityViewModel mViewModel;
     
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_my);
     
            mViewModel = ViewModelProviders.of(this).get(MyActivityViewModel.class);
     
            mViewModel.getProductShowLiveData().observe(this, product -> {
                if (product.isConnected()) {
                    // 更新UI
                    mTextView.setText(getString(product.getNameId());
                }
            });
            // 一些业务逻辑
            mViewModel.doSomeBusiness();
        }
    }
    
    public class MyActivityViewModel extends BaseRxViewModel<MyActivityRepository> {
       
        private MutableLiveData<ProductTypeShowData> mProductShowLiveData = new MutableLiveData<>();
     
        public MyActivityViewModel(@NonNull Application application) {
            super(application);
        }
     
        private void doSomeBusiness() {
            // 一些业务逻辑
            ...
            // 通知UI更新,并将UI需要的数据源setValue通知出去
            mProductShowLiveData.setValue(XXXX);
        }
    }
    

    二. 原理

    ViewModel

    ViewModel的功能在于能自动处理转屏,切换输入法等配置变更引起的Activity销毁,生命周期重走问题,使得重新生成的Activity拿到的依然是之前的ViewModel,也即该ViewModel的生命周期是从Activity start到finish,中间不管因为配置变更Activity销毁重启多少次,ViewModel都是不会重启的(所以不能在ViewModel里持有Activity的Context)

    ViewModel生命周期
    具体源码分析推荐可以阅读这篇文章,基本原理是在Activity上add一个setRetainInstance(true)的HolderFragment,由其保存对应Activity的ViewModelStore,在Activity重走生命周期后,在onCreate()中再次通过ViewModelProviders.of(this)找到对应的HolderFragment的ViewModelStore,并通过get(xxxViewModel.class)以class为key,找到存在ViewModelStore中的ViewModel。

    (故在Fragment之间可以通过ViewModelProviders.of(getActivity()).get(xxxViewModel.class)来获取到相同的ViewModel,从而实现Fragment之间的ViewModel共享)


    ViewModelProviders.of(this).get(xxxViewModel.class)实际的内部结构

    LiveData

    LiveData的功能在于,在于setValue时(非UI线程使用postValue),会根据当前界面的生命周期(通过在界面上add一个无界面的ReportFragment,接收其生命周期回调来实现,很经典的做法,Glide比起其他的图片加载库有感知生命周期的功能也是这么做的),选择要不要回调到UI界面更新(在对应的Activity调用onStart()之前,onStop()之后,都是不会更新界面的),那么如果遇到这些情况,LiveData会选择在每一次对应界面的生命周期变化时,以及调用LiveData.observe()建立监听关系时再去尝试回调UI更新


    LiveData分发消息过程以及触发时机 Lifecycle.State

    关于这块的源码,代码量其实也不多,LiveData相关的还是比较简单的,LifecycleRegistry分发生命周期稍微有一点点绕,但也还好,在使用过的基础上一天基本就能通读代码

    源码解析链接里这位博主写的很好:

    LiveData:

    https://jekton.github.io/2018/07/14/android-arch-LiveData/

    Lifecycle

    https://jekton.github.io/2018/07/06/android-arch-lifecycle/

    三. 总结

    LiveData,ViewModel以及Lifecycle相关的api都已经被收录进了google推出的帮助开发者更快更专注地开发app的jetpack火箭背包里,现在最新的代码都在androidx库上更新

    https://developer.android.com/topic/libraries/architecture/adding-components#lifecycle

    AndroidX库引入

    相关文章

      网友评论

        本文标题:LiveData,ViewModel,Lifecycle原理解析

        本文链接:https://www.haomeiwen.com/subject/nkjzdqtx.html