LiveData,ViewModel,Lifecycle原理解析以及使用

一. 使用

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)

4086245-c8a5a6e8e36f4587.png
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共享)


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

LiveData

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


4086245-7aa7795bbedb0d7c.png
LiveData分发消息过程以及触发时机
4086245-ca4acdc32fe4d0cd.png
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

4086245-5e9983b4ad15021a.png
AndroidX库引入

猜你喜欢

转载自blog.csdn.net/weixin_34008933/article/details/86915509