美文网首页JetPack学习笔记
JetPack学习笔记之ViewModel和LiveData

JetPack学习笔记之ViewModel和LiveData

作者: 秀儿2020 | 来源:发表于2020-08-06 14:41 被阅读0次

    在页面功能较简单的情况下,通常将UI的交互、数据获取等业务全部写在页面中,即通常的MVC模式。但是在页面功能较复杂的情况下,这样做不合适,因为它不符合单一职责原则,页面只应该负责处理用户与UI的交互部分,并将数据展示到页面上,与数据相关的业务逻辑应该单独处理和存放。

    Android提供了ViewModel类,专门用于存放应用程序页面所需要的数据,他是介于View(视图)和Model(数据模型)之间的一个东西,起到了桥梁的作用,使得视图和数据既能够分开,又能够保持通信。

    ViewModel中不仅可以用于存放数据,还可以在其中存放一些数据相关的逻辑,因此,ViewModel中的数据随着业务的变化而变化。

    对于页面来说,他并不关心ViewModel中的业务逻辑,他只关心数据是什么,并且希望在数据发生变化的时候可以收到通知以做出更新。这就是LiveData(活着的数据)的作用,LiveData是一个可被观察的数据容器类,即ViewModel中的数据发生变化时,通知页面。因此LiveData通常被存放在ViewModel中,用于包装那些需要被外界观察的数据。

    1、LiveData 的基本使用方法。

    LiveData是一个抽象类,通常我们使用的是他的直接子类MultableLiveData。

    /**
     * 项目名称 JetPackPro
     * 创建人 xiaojinli
     * 创建时间 2020/8/6 2:13 PM
     **/
    public class TimerViewModel extends ViewModel {
        private MutableLiveData<Integer> mutableLiveData;
    
        public MutableLiveData<Integer> getMutableLiveData(){
            if(mutableLiveData == null){
                mutableLiveData = new MutableLiveData<>();
            }
            return mutableLiveData;
        }
        
        //变更数据
        public void changeValue() {
            mutableLiveData.setValue(2);
        }
    }
    
    2、定义完ViewModel和LiveData之后,如何进行数据通信呢?
    public class TimerActivity extends AppCompatActivity {
        private static final String TAG = "TimerActivity";
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_timer);
            initComponent();
        }
    
    
        private void initComponent() {
            //通过ViewModelProvider得到ViewModel
            TimerViewModel timerViewModel = new ViewModelProvider(this).get(TimerViewModel.class);
            //得到ViewModel中的LiveData
            final MutableLiveData<Integer> liveData = timerViewModel.getMutableLiveData();
            //通过LiveData中的observer观察ViewModel中数据的变化
            liveData.observe(this, new Observer<Integer>() {
                @Override
                public void onChanged(Integer integer) {
                    Log.d(TAG,"当前值为: " + integer);
                }
            });
            //更新数据
            timerViewModel.changeValue();
        }
    }   
    

    运行结果:

    2020-08-06 14:20:34.058 20266-20266/com.example.jetpackpro D/TimerActivity: 当前值为: 2
    

    通过LiveData 的observer方法对LiveData包装的数据进行观察,可以看出,在changeValue之后,可以在页面中观察到数据的改变,以便进行后续操作。

    以上在ViewModel中更新数据使用的是setValue方法,该方法用于在UI线程对数据进行更新,如果要在非UI线程对数据进行更新,可以使用postValue方法,以下是使用postValue方法更新数据时的ViewModel。

    /**
     * 项目名称 JetPackPro
     * 创建人 xiaojinli
     * 创建时间 2020/8/6 2:13 PM
     **/
    public class TimerViewModel extends ViewModel {
        private MutableLiveData<Integer> mutableLiveData;
    
        public MutableLiveData<Integer> getMutableLiveData(){
            if(mutableLiveData == null){
                mutableLiveData = new MutableLiveData<>();
            }
            return mutableLiveData;
        }
    
    
        public void changeValue() {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for(int i=0;i<10;i++){
                        try {
                            Thread.sleep(1000);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        mutableLiveData.postValue(i);
                    }
    
                }
            }).start();
        }
    }
    

    运行结果:

    2020-08-06 14:25:35.722 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 0
    2020-08-06 14:25:36.741 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 1
    2020-08-06 14:25:37.751 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 2
    2020-08-06 14:25:38.756 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 3
    2020-08-06 14:25:39.760 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 4
    2020-08-06 14:25:40.762 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 5
    2020-08-06 14:25:41.766 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 6
    2020-08-06 14:25:42.773 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 7
    2020-08-06 14:25:43.776 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 8
    2020-08-06 14:25:44.780 20529-20529/com.example.jetpackpro D/TimerActivity: 当前值为: 9
    

    在页面中会收到每次数据的更新结果。

    因为ViewModel能够从页面(Activity)中剥离出来,独立于页面的变化,即使旋转屏幕导致Activity重建,也不会影响到ViewModel,只要Activity不被销毁,ViewModel就会一直存在。基于ViewModel 的这个特性,我们可以实现Activity中多个Fragment之间的通信。因为Fragment可以看做是Activity 的子页面,即Fragment即使彼此独立,但是都属于同一个Activity。所以只要定义两个Fragment,让其绑定同一个ViewModel,并在其内实现对ViewModel数据的监听,只要在其中一个Fragment中触发ViewModel数据的更新,另一个Fragment中就会收到监听事件,从而可以实现两个Fragment之间的通信。

    LiveData的本质是观察者模式,并且能够感知页面的生命周期,只有在页面存活时才会进行通知,从而避免了内存泄露。

    但是存在一个例外,如果使用LiveData 的observerForever方法,当LiveData包装的数据发生变化时,不管页面处于什么状态,都会触发通知,他是忽略了页面的生命周期的,所以存在内存泄露的方法,在用完时,可以通过removeObserver方法移除监听。

    LiveData不仅可以用在ViewModel中,在其他地方也可以发挥作用,比如Room数据组件中。

    参考《Android Jetpack应用指南》

    相关文章

      网友评论

        本文标题:JetPack学习笔记之ViewModel和LiveData

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