JetPack学习笔记之ViewModel和LiveData

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

推荐阅读更多精彩内容