Android组件 LiveData与MutableLiveData教程

2020-11-22 23:04:39

参考地址 Android开发 LiveData与MutableLiveData详解

前言

  LiveData与ViewMode是经常搭配在一起使用的,但是为了不太混乱,我还是拆分开来说明,此篇博客只讲解 LiveData 与 MutableLiveData的概念与使用方式(但是会涉及到ViewMode的部分代码).

LiveData是干什么的?

  由于LiveData和MutableLiveData都是一个概念的东西(只是作用范围不同)所以就不重复解释了,直接理解LiveData就可以明白MutableLiveData

  直接理解LiveData的字面意思是前台数据,其实这其实是很准确的表达.下面我们来说说LiveData的几个特征:

1.首先LiveData其实与数据实体类(POJO类)是一样的东西,它负责暂存数据.

2.其次LiveData其实也是一个观察者模式的数据实体类,它可以跟它注册的观察者回调数据是否已经更新.

3.LiveData还能知晓它绑定的Activity或者Fragment的生命周期,它只会给前台活动的activity回调(这个很厉害).这样你可以放心的在它的回调方法里直接将数据添加到View,而不用担心会不会报错.(你也可以不用费心费力判断Fragment是否还存活)

LiveData与MutableLiveData区别

LiveData与MutableLiveData的其实在概念上是一模一样的.唯一几个的区别如下:

1.MutableLiveData的父类是LiveData

2.LiveData在实体类里可以通知指定某个字段的数据更新.

3.MutableLiveData则是完全是整个实体类或者数据类型变化后才通知.不会细节到某个字段

 

LiveData简单使用Demo

创建LiveData

复制代码

public class DemoData extends LiveData<DemoData> {    private int tag1;    private int tag2;    
    public int getTag1() {        return tag1;

    }    public void setTag1(int tag1) {        this.tag1 = tag1;
        postValue(this);
    }    public int getTag2() {        return tag2;
    }    public void setTag2(int tag2) {        this.tag2 = tag2;
        postValue(this);
    }
}

复制代码

很简单,只要继承LiveData并且在泛型里写下你的实体类,唯一需要注意的,postValue(this);这个方法是用于回调数据更新的方法. 你可以在你需要被观察的数据里添加.

创建ViewModel

我们需要在ViewModel实例化DemoData这个类. ViewModel(这个会在另一篇博客介绍)这个是用于管理多个Activity或者Fragment数据的类。ViewModel是MVVM的概念。你可以百度一下,google提供这套东西就是为了MVVM。

复制代码

public class DemoViewModel extends ViewModel {    // TODO: Implement the ViewModel
    private DemoData mDemoData = new DemoData();    public DemoData getDemoData() {        return mDemoData;
    }
}

复制代码

在Activity或者Fragment绑定

复制代码

public class Demo2Activity extends AppCompatActivity {    private static final String TAG = "Demo2Activity";    private Button mBtnAddData;    private DemoViewModel mDemoViewModel;

    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_demo2);
        mBtnAddData = findViewById(R.id.btn_add_data);
        mDemoViewModel = ViewModelProviders.of(this).get(DemoViewModel.class);//获取ViewModel,让ViewModel与此activity绑定
        mDemoViewModel.getDemoData().observe(this, new Observer<DemoData>() { //注册观察者,观察数据的变化            @Override            public void onChanged(DemoData demoData) {
                Log.e(TAG, "onChanged: 数据有更新");
            }
        });
        
        mBtnAddData.setOnClickListener(new View.OnClickListener() {
            @Override            public void onClick(View v) {
                Log.e(TAG, "onClick: 已经点击");
                mDemoViewModel.getDemoData().setTag1(123); //这里手动用按键点击更新数据
            }
        });
    }
}

复制代码

当我们点击按键后就会有数据更新后的回调触发:

2019-09-18 19:45:53.821 6649-6649/demo.yt.com.demo E/Demo2Activity: onClick: 已经点击2019-09-18 19:45:53.824 6649-6649/demo.yt.com.demo E/Demo2Activity: onChanged: 数据有更新

前面提过了,但是这里还是需要重新提一下! 注意! 这个数据只给前台的活动回调.

 

MutableLiveData简单使用Demo

   前面已经解释了,所以我们这边直接看代码

创建MutableLiveData

复制代码

public class DemoViewModel extends ViewModel {    // TODO: Implement the ViewModel
    private MutableLiveData<String> myString = new MutableLiveData<>();    public MutableLiveData<String> getMyString(){        return myString;
    }    public void setMyString(String string) {        this.myString.setValue(string);
    }
}

复制代码

因为MutableLiveData只是作用于变量所以我们直接就可以在ViewModel里实例化它,并且在泛型里标注变量的类型.

复制代码

public class MutableLiveData<T> extends LiveData<T> {
    @Override    public void postValue(T value) {        super.postValue(value);
    }

    @Override    public void setValue(T value) {        super.setValue(value);
    }
}

复制代码

 

在Activity或者Fragment绑定

复制代码

public class Demo1Activity extends AppCompatActivity {    private static final String TAG = "Demo1Activity";    private DemoViewModel mDemoViewModel;    private Button mBtn1;

    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_demo);
        mBtn1 = findViewById(R.id.btn_1);
        mDemoViewModel = ViewModelProviders.of(this).get(DemoViewModel.class);//获取ViewModel,让ViewModel与此activity绑定
        mDemoViewModel.getMyString().observe(this, new Observer<String>() { //注册观察者            @Override            public void onChanged(String s) {
                Log.e(TAG, "onChanged: 值有变化="+s);
            }
        });

        mBtn1.setOnClickListener(new View.OnClickListener() {
            @Override            public void onClick(View v) {
                mDemoViewModel.setMyString("测试"); //用手动按键点击改变值

            }
        });
    }
}

复制代码

 结果:

2019-09-18 19:59:38.294 6961-6961/demo.yt.com.demo E/Demo1Activity: onChanged: 值有变化=测试

API全解

postValue()

  可能你已经在上面看到几次调用此方法了。postValue的特性如下:

  1.此方法可以在其他线程中调用

  2.如果在主线程执行发布的任务之前多次调用此方法,则仅将分配最后一个值。

  3.如果同时调用 .postValue(“a”)和.setValue(“b”),一定是值b被值a覆盖。

setValue()

  setValue()的特性如下:

  1.此方法只能在主线程里调用

getValue()

  返回当前值。 注意,在后台线程上调用此方法并不能保证将接收到最新的值。

removeObserver(@NonNull final Observer<? super T> observer) 

  移除指定的观察者

例子:

复制代码

        Observer<String> observer = new Observer<String>() {
            @Override            public void onChanged(String s) {
                mText.setText("内容改变=" + s);
            }
        };
        mMainViewModel.getContent().observe(this, observer);//绑定
        mMainViewModel.getContent().removeObserver(observer);//解除

复制代码

removeObservers(@NonNull final LifecycleOwner owner)

  移除当前Activity或者Fragment的全部观察者

mMainViewModel.getContent().removeObservers(this);

 hasActiveObservers()

  如果此LiveData具有活动(Activity或者Fragment在前台,当前屏幕显示)的观察者,则返回true。其实如果这个数据的观察者在最前台就返回true,否则false。

hasObservers()

  如果此LiveData具有观察者,则返回true。

 

observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer)

  设置此LiveData数据当前activity或者Fragment的观察者,会给此activity或者Fragment在前台时回调数据。

 

observeForever(@NonNull Observer<? super T> observer)

  1.设置永远观察者,永远不会被自动删除。您需要手动调用removeObserver(Observer)以停止观察此LiveData,

  2.设置后此LiveData,一直处于活动状态,不管是否在前台哪里都会获得回调。

 


  • 2020-03-17 09:47:05

    video标签视频不自动播放的问题

    添加 muted 属性,就可以通过地址栏进入网页的时候自动播放了,手机端还是有的有限制的,比如iphone浏览器,就不行,苹果手机为了保护用户的流量和用户的意愿,是禁止自动播放的,必须有手动触发。

  • 2020-03-17 14:21:31

    nuxt+pm2 自动化部署及打包后文件自动上传阿里云 oss(精华)

    部署nuxtjs,这一篇文章就够了,pm2 代码自动发布依赖于 git 工具,先将 ssh 密钥配置再你的代码仓库(github 或者 gitLab),具体操作自行 google 或者点击github 配置 ssh。 使用 ssh 密钥链接服务器 s $ ssh-copy-id root@1.2.3.4 # 把本机的 SSH 秘钥添加至服务器,配置成功后,以后就不需要再执行这条 SSH 命令了

  • 2020-03-18 21:15:34

    使用canvas画布解决百度地图自定义图层全球连续显示问题

        基于百度地图的Web API进行自定义图层叠加时,默认的图层只能叠加到全球范围以内,即经度范围为[-180, 180],而无法将图层叠加到默认的全球范围以外,即经度范围超出了[-180, 180]之后,经纬度坐标会自动回归到(0, 0),而导致在地图拖拽时全球以外无法连续显示想要的图层,此时可以基于百度地图的自定义图层将经纬度坐标转为像素点使用画布canvas来解决该问题。解决后效果如下图所示: ———————————————— 版权声明:本文为CSDN博主「宏伟杰作」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.csdn.net/u011284073/article/details/80549950

  • 2020-03-18 21:18:01

    node-canvas实现百度地图个性化底图绘制

    随着nodejs的推出,node的并发和异步的强大能力,越来越多的得到应用,而且取得了非常不错的效果。 作为一个前端工程师对node.js自然有着一份更深的感情,跃跃欲试的心情,总希望能将它应用到产品中来。

  • 2020-03-18 21:19:28

    高德地图和canvas画图结合应用的一些感想(一)

    入了团队才发现,该项目前后端分离,后端工程师已就位主要实现接口,IOS端工程师也已就位,还差一个web前端工程师。背脊一凉,我之前虽然写过一些js和css,虽有点功底但是离前端工程师还是有距离的啊。在和朋友说明情况后,朋友也是胆大,让我试试,主要他实在找不到人了(也有可能目前前端工程师报价都太贵了,创业嘛,能节约就节约,能理解。。。),没办法,走一步算一步吧。