LiveData使用与解析

上一篇文章ViewModel简单使用与解析我们说了ViewModel相关的知识,接下来我们要说的是ViewModel的终极搭档LiveData。
先来说一下LiveData的优点:

  • 减少内存泄漏,不需要手动处理生命周期,
    LiveData会绑定实现了LifecycleOwner接口的对象,并会在这些对象被销毁后自行清理。
    Activity/Fragment都继承并实现了这个接口,所以我们只需对数据进行订阅便可,不需要手动 去停止或恢复观察。(也可以使用observeForever(Observer)方法注册一个没有关联LifecycleOwner的对象,但是这得手动调用removeObserver(Observer)方法进行移除)
  • 在最合适的时候通知更新数据
    LiveData只会将数据通知给处于活跃状态下的对象,因此可以规避很多异步任务结束后更新已经被回收的UI而造成的空指针异常,而非活跃状态下的对象当基状态变成活跃时也会收到最新的数据。
  • 正确应对Configuration Change
    如果一个Activity或Fragment由于Configuration Change更改(如设备旋转)而重新创建,它会立即收到最新的可用数据。
  • 共享资源
    您可以使用单例模式扩展LiveData对象并包装成系统服务,以便在应用程序中进行共享。LiveData对象一旦连接到系统服务,任何需要该资源的Observer都只需观察这个LiveData对象。 有关更多信息,请参阅扩展LiveData

1.创建LiveData对象

public class MViewModel extends ViewModel {

    MutableLiveData<String> mString;

    public MutableLiveData<String> getString(){
        if(mString==null){
            mString=new MutableLiveData<>();
        }
        return mString;
    }

    public MutableLiveData<String> getMsgString(){
        if(msgString==null){
            msgString=new MutableLiveData<>();
        }
        return msgString;
    }
}

一般都是在ViewModel中声明和保存LiveData,这有利于:

  • 保持Activity/Fragment的精简性,这些UI控制器负责显示数据而不是保存数据状态。
  • 将LiveData实例与特定Activity/Fragment实例分离,这将使得LiveData对象在发生Configuration Change后仍然存活。

2.订阅LiveData对象

官方推荐在应用程序组件的onCreate()方法开始观察LiveData对象,理由是:

  • 确保系统不会从Activity或Fragment的onResume()方法中进行多余的调用。
  • 确保Activity或Fragment一旦变为活动状态时,就有可展示的数据。 当应用程序组件处于STARTED状态,它就需从它所观察的LiveData对象中接收到最新的值。 所以我们需要在一开始就设置好订阅。
protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mTextView=findViewById(R.id.desc);
        mViewModel = ViewModelProviders.of(this).get(MViewModel.class);
        mViewModel.getString().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                Log.e("MainActivity", "耗时任务结束返回数据");
                mTextView.setText(s);
            }
});

3.更新LiveData对象

一般在ViewModel中对数据进行处理,当我们拿到数据后将值赋值给LiveData

    public void startTask(){
        new Thread(){
            @Override
            public void run() {
                //请求网络数据、数据库、加载大图等。
                //如果在Activity转屏的时候取消这些任务,那恢复的时候就要重新加载,势必浪费资源
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                //更新LiveData
                mString.postValue("我是来自3秒后的数据");
                super.run();
            }
        }.start();
    }

setValue()或者postValue()都可以用来更新LiveData的值,区别在于:
setValue要在主线程中使用,postValue在后台线程中使用。

4.LiveData Transformations

有时候我们想在ViewModel中对LiveData进行数据转换之后再输出给observer,比如我我拿到了用户ID,我们需要通过ID拿到用户信息再去更新UI,那我们就可以用Transformations来对一个LiveData对象进行转换,Transformations类目前只提供两个转换方法map()和switchMap():

 static <X, Y> LiveData<Y> map(LiveData<X> source,Function<X, Y> mapFunction) {}
 static <X, Y> LiveData<Y> switchMap(LiveData<X> source,Function<X, LiveData<Y>> switchMapFunction){}

简单对比一下你会发现这两个方法只是最后一个传入的参数不同而已,返回结果跟第一个参数是一样的,至于选择哪个方法,就完全是看你要传入的参数类型了

    LiveData<Long> userId...;
    LiveData<UserBean> userBean=Transformations.map(userId, new Function<Long, UserBean>() {
        @Override
        public UserBean apply(Long input) {
            //比如说你刚好有个方法是通过userId获取到userBean,并且是返回UserBean类型的就用map
            return new UserBean();
        }
    });
    LiveData<UserBean> userBean=Transformations.switchMap(userId, new Function<Long, LiveData<UserBean>>() {
        @Override
        public LiveData<UserBean> apply(Long input) {
            //比如说你刚好有个方法是通过userId获取到userBean,并且是返回LiveData<UserBean>类型的就用switchMap
            return new MutableLiveData<UserBean>(){};
        }
    });

再来看下该类完整的代码

public class Transformations {

    private Transformations() {
    }

    @MainThread
    public static <X, Y> LiveData<Y> map(@NonNull LiveData<X> source,
            @NonNull final Function<X, Y> mapFunction) {
        final MediatorLiveData<Y> result = new MediatorLiveData<>();
        result.addSource(source, new Observer<X>() {
            @Override
            public void onChanged(@Nullable X x) {
                result.setValue(mapFunction.apply(x));
            }
        });
        return result;
    }

    @MainThread
    public static <X, Y> LiveData<Y> switchMap(
            @NonNull LiveData<X> source,
            @NonNull final Function<X, LiveData<Y>> switchMapFunction) {
        final MediatorLiveData<Y> result = new MediatorLiveData<>();
        result.addSource(source, new Observer<X>() {
            LiveData<Y> mSource;

            @Override
            public void onChanged(@Nullable X x) {
                LiveData<Y> newLiveData = switchMapFunction.apply(x);
                if (mSource == newLiveData) {
                    return;
                }
                if (mSource != null) {
                    result.removeSource(mSource);
                }
                mSource = newLiveData;
                if (mSource != null) {
                    result.addSource(mSource, new Observer<Y>() {
                        @Override
                        public void onChanged(@Nullable Y y) {
                            result.setValue(y);
                        }
                    });
                }
            }
        });
        return result;
    }
}

你会发现其实就是通过类把LiveData<X>转换成了MediatorLiveData<Y>输出。

5. MediatorLiveData

我们前面有提到过,只有LifecycleOwner对象才能使用observer方法对LiveData进行监听,而Transformations的转换思路显然是通过对上一个LiveData进行了监听然后在其值发生变化的时候封装成MediatorLiveData对象输出。所以应该是用到了LiveData的observeForever方法。
我们先来看下它的addSource方法

    public <S> void addSource(@NonNull LiveData<S> source, @NonNull Observer<? super S> onChanged) {
        Source<S> e = new Source<>(source, onChanged);
        Source<?> existing = mSources.putIfAbsent(source, e);
        if (existing != null && existing.mObserver != onChanged) {
            throw new IllegalArgumentException(
                    "This source was already added with the different observer");
        }
        if (existing != null) {
            return;
        }
        if (hasActiveObservers()) {
            e.plug();
        }
    }

传入的LiveData和Observer被用来构造Source对象,看下Source代码

private static class Source<V> implements Observer<V> {
        final LiveData<V> mLiveData;
        final Observer<? super V> mObserver;
        int mVersion = START_VERSION;

        Source(LiveData<V> liveData, final Observer<? super V> observer) {
            mLiveData = liveData;
            mObserver = observer;
        }

        void plug() {
            //此处完成了对传入的LiveData的订阅
            mLiveData.observeForever(this);
        }

        void unplug() {
            mLiveData.removeObserver(this);
        }

        @Override
        public void onChanged(@Nullable V v) {
            if (mVersion != mLiveData.getVersion()) {
                mVersion = mLiveData.getVersion();
                mObserver.onChanged(v);
            }
        }
    }

果然是用到了observeForever对传入的LiveData进行了监听,然后在onChanged方法将值继续传递下去。而MediatorLiveData可以多少调用addSource,其内部用了一个Map来存储构造生成的Source对象。
MediatorLiveData相当于一个自定义的LiveData,其内部定义了一个Map用于存储

private SafeIterableMap<LiveData<?>, Source<?>> mSources = new SafeIterableMap<>();

所以MediatorLiveData除了用于上面Transformations的转换操作之外还可用来做合并订阅的操作。

 * LiveData<Integer> liveData1 = ...;
 * LiveData<Integer> liveData2 = ...;
 *
 * MediatorLiveData<Integer> liveDataMerger = new MediatorLiveData<>();
 * liveDataMerger.addSource(liveData1, value -> liveDataMerger.setValue(value));
 * liveDataMerger.addSource(liveData2, value -> liveDataMerger.setValue(value));

这样只要是liveData1、liveData2其中一个数据的变化都会引发liveDataMerger去发出更新数据的通知。

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

推荐阅读更多精彩内容