Android LiveData使用注意事项
关于LiveData是什么以及基本使用方式,请参考官方文档:https://developer.android.com/topic/libraries/architecture/livedata?hl=zh-cn。
简单来说,LiveData是一个可被观察的数据容器类。它将数据包装起来,使得数据成为“被观察者”,页面成为“观察者”。当ViewModel存放页面所需要的各种数据发生变化时,通过LiveData的方式实现对页面的通知,完成ViewModel与页面组件之间的通信。
那么在使用时发现有以下几个地方需要注意:
1.回调通知
LiveData的观察者会在每次进入活跃态时收到回调,比如从后台回到前台,界面重启等等。如果只想收到一次回调的话,可以使用SingleLiveEvent。要注意SingleLiveEvent仅限于一个观察者。如果添加了多个则只会调用一个,并且不能保证哪一个。
2.数据倒灌
所谓数据倒灌是一种形象的说法,它是指先setValue/postValue,后调用observe(new Obs()),至此收到了回调。然后再调用observe(new anotherObs()),如果还能收到第一次的回调,也就是旧数据。解决方案可以参考开源项目:UnPeek-LiveData。
3.事件包装
上面提到SingleLiveEvent仅限于一个观察者,如果需要多个观察者该如何处理呢,方案就是使用事件包装。定义一个数据包装器,内部判断事件是否消费了,被消费后则不再回调通知。代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
/**
* 事件包装器,明确地管理事件是否已经被处理
* @param <T> ViewModel中的数据,比如:
* MutableLiveData<LiveEventWrapper<String>>()
*/
public class LiveEventWrapper<T> {
private T content;
private boolean hasBeenHandled;
public LiveEventWrapper(T content) {
this.content = content;
}
/**
* Returns the content and prevents its use again.
*/
public T getContentIfNotHandled() {
if (hasBeenHandled) {
return null;
} else {
hasBeenHandled = true;
return content;
}
}
/**
* Returns the content, even if it's already been handled.
*/
public T peekContent() {
return content;
}
public boolean isHasBeenHandled() {
return hasBeenHandled;
}
}
或者
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
import androidx.annotation.NonNull;
import androidx.lifecycle.LifecycleOwner;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.Observer;
public class CleanLiveData<T> extends LiveData<T> {
private boolean hasModified = false;
@Override
public void observe(@NonNull LifecycleOwner owner, @NonNull final Observer<? super T> observer) {
super.observe(owner, new Observer<T>() {
private boolean hasIntercept = false;
@Override
public void onChanged(T t) {
if (!hasModified || hasIntercept) {
observer.onChanged(t);
}
hasIntercept = true;
}
});
}
@Override
public void observeForever(@NonNull final Observer<? super T> observer) {
super.observeForever(new Observer<T>() {
private boolean hasIntercept = false;
@Override
public void onChanged(T t) {
if (!hasModified || hasIntercept) {
observer.onChanged(t);
}
hasIntercept = true;
}
});
}
@Override
public void setValue(T value) {
super.setValue(value);
hasModified = true;
}
@Override
public void postValue(T value) {
super.postValue(value);
hasModified = true;
}
}
总结来看,以上现象基本都是由于LiveData的粘性特性引发的,因此在使用LiveData的时一定要搞清楚它的概念和原理。
本文由作者按照 CC BY 4.0 进行授权