一般情况下LiveData都是搭配这ViewModel使用,这里先介绍一下LiveData,再结合ViewModel使用
1、什么是LiveData
2、什么是MutableLiveData
3、LiveData与MutableLiveData区别
4、LiveData的使用
5、MutableLiveData的使用
6、LiveData和MutableLiveData的可变与不可变
7、其他方法
public class UserBean extends LiveData<UserBean> {
String name;
int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
postValue(this);
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
postValue(this);
}
}
postValue(this);这个方法是用于触发回调数据更新的方法. 你可以在你需要被观察的数据里添加.
小注意点:如果不使用postValue(this)会怎么样?
public void setAge(int age) {
this.age = age;
// postValue(this);
// 如果不写的话,单独调用setAge后,在外getAge获取不到值,因为回调没有被触发,
// 如果name写了postValue(this),更新了name也会连着age更新,因为传的是this
}
你也可以集成LiveData<String>,那么postValue(string),回调中只会接收postValue最后的赋值。
public class MyViewModel extends ViewModel {
public UserBean mUserBean = new UserBean();
public UserBean getUserBean() {
return mUserBean;
}
}
//通过ViewModeProvider 把activity和ViewModel绑定起来。
myViewModel = new ViewModelProvider(this, new ViewModelProvider.NewInstanceFactory()).get(MyViewModel.class);
observer = new Observer<UserBean>() {
@Override
public void onChanged(UserBean userBean) {
Log.e("activity",userBean.getName());
mBinding.textShow.setText("name="+userBean.getName()+" ,, age="+userBean.getAge()+"");
}
};
//绑定观察者
myViewModel.getUserBean().observe(this, observer);
a = 0;
mBinding.btnUpdata.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
myViewModel.getUserBean().setName("大王"+a);
myViewModel.getUserBean().setAge(a++);
}
});
注意:这里的myViewModel.getUserBean().observe() 是LiveData的方法。
我们上面创建了一个Observer,并且和activity关联,然后通过一个按钮去动态设置值,果然回调里就会有数据更新。
public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
myViewModel.getUserBean().removeObserver(observer);
它是直接作用在变量上,mStr.setValue(s);触发观察者的回调。
public class MyViewModel extends ViewModel {
private MutableLiveData<String> mStr;
public MutableLiveData<String> getmStr() {
return mStr;
}
public void setmStr(String s) {
if(mStr==null){
mStr = new MutableLiveData<>();
}
mStr.setValue(s);
}
}
myViewModel.getmStr().observe(this, new Observer<String>() {
@Override
public void onChanged(String s) {
}
});
其实就是setValue和postValue是否暴露问题,LiveData中是不支持外部使用,而MutableLiveData特意放开
@SuppressWarnings("WeakerAccess")
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);
}
}
可以看到在MutableLiveData包裹的有setValue,postValue方法
当然我们也可以让它不对外暴露setValue,postValue方法,返回值修改为LiveData<String>
public class MyViewModel extends ViewModel {
private MutableLiveData<String> mStr= new MutableLiveData<>();
public LiveData<String> getmStr() {
return mStr;
}
//在外面只能通过这种方式去更新
public void setmStr(String s) {
mStr.setValue(s);
}
}
postValue()
postValue的特性如下:
1.此方法可以在其他线程中调用
2.如果在主线程执行发布的任务之前多次调用此方法,则仅将分配最后一个值。
3.如果同时调用 .postValue()和.setValue(),结果是post的值覆盖set值。
setValue()
setValue()的特性如下:
1.此方法只能在主线程里调用
getValue()
返回当前值。 注意,在后台线程上调用此方法并不能保证将接收到最新的值。
removeObserver(@NonNull final Observer<? super T> observer)
移除指定的观察者
removeObservers(@NonNull final LifecycleOwner owner)
移除当前Activity或者Fragment的全部观察者
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,一直处于活动状态,不管是否在前台哪里都会获得回调。
LiveData一般使用在实体类,MutableLiveData作用在变量上,他们通常和ViewModel结合使用,上面例子过于简单,工作当中可按照业务需要进行调整。