如何使用一个可观察值的状态来跳过另一个可观察值的值?
How to use the state of one observable to skip values of another observable?
最好用一个简短的例子来解释。
假设这是我要过滤的可观察源
Observable.interval(1, TimeUnit.SECONDS)
我使用复选框来处理过滤器状态。未选中该框时,我想跳过所有值。
我使用 RxAndroid 来获取复选框的可观察值,如下所示:
RxCompoundButton.checkedChanges(checkBox)
这是我的代码:
Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> {
if (!isChecked) {
return null;
} else {
return intervalCounter;
}
}
).filter(Objects::nonNull)
我得到了想要的输出
interval 1--2--3--4--5--6--7--8--9
checkbox 0-----1--------0-----1---
combineLatest
result ------3--4--5--------8--9
我刚开始使用 RxJava,我的 "solution" 感觉不对,因为:
可以看到combine函数returns一个神奇的值null
然后filter会跳过这些nulls
.
这意味着即使我已经知道要跳过此数据,也会调用过滤函数。
- 也许我应该使用另一个运算符
- 有没有办法在过滤器函数中使用可观察复选框
- 也许在组合函数中有一些方法可以表明我想跳过这个数据
你想要输出的图片不正确。在您的实现中,combineLatest
将 null
组合到流中:
interval 1--2--3--4--5--6--7--8--9
checkbox 0-----1--------0-----1---
combineLatest N--N--3--4--5--N--N--8--9
filter(NonNull)------3--4--5--------8--9
恕我直言,在 Rx Stream 中使用 null
作为信号并不好,开发人员很容易陷入 NullPointerException
.
要避免使用null
,有两个措施。第一个是将结果转换为 Pair,然后再应用 filter & map。
很简单Pair
Class:
public class Pair<T, U> {
T first;
U second;
public Pair(T first, U second) {
this.first = first;
this.second = second;
}
}
那么整个实现是这样的:
Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> new Pair<>(isChecked,intervalCounter)
).filter(pair -> pair.first)
.map(pair -> pair.second) // optional, do this if you only need the second part
数据流向:
interval 1 2 3 4
| | | |
checkbox F | T | F | T F
| | | | | | |
combineLatest F1 T1 T2 F2 F3 T3 F4
| | |
filter(first=T) T1 T2 T3
| | |
map(second) 1 2 3
或者,如果您可以在您的项目中 Java 8,请使用 Optional
来避免 null
,这与您的解决方案非常相似,但它让其他开发人员意识到流信号是 Optional
.
Observable<Optional<Integer>> o1 = Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> {
if (!isChecked) {
return Optional.empty();
} else {
return Optional.of(intervalCounter);
}
}
)
Observable<Integer> o2 = o1.filter(Optional::isPresent).map(Optional::get)
一些更简单的方法,与 combineLatest
不同,但与您想要的结果图片相同。
// Approach 1
Observable.interval(1, TimeUnit.SECONDS).filter( i -> checkBox.isEnabled())
// Approach 2
Observable.interval(1, TimeUnit.SECONDS)
.withLatestFrom(
RxCompoundButton.checkedChanges(checkBox),
(isChecked, intervalCounter) -> {
if (!isChecked) {
return Optional.empty();
} else {
return Optional.of(intervalCounter);
}
}).filter(Optional::isPresent).map(Optional::get)
最好用一个简短的例子来解释。
假设这是我要过滤的可观察源
Observable.interval(1, TimeUnit.SECONDS)
我使用复选框来处理过滤器状态。未选中该框时,我想跳过所有值。
我使用 RxAndroid 来获取复选框的可观察值,如下所示:
RxCompoundButton.checkedChanges(checkBox)
这是我的代码:
Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> {
if (!isChecked) {
return null;
} else {
return intervalCounter;
}
}
).filter(Objects::nonNull)
我得到了想要的输出
interval 1--2--3--4--5--6--7--8--9
checkbox 0-----1--------0-----1---
combineLatest
result ------3--4--5--------8--9
我刚开始使用 RxJava,我的 "solution" 感觉不对,因为:
可以看到combine函数returns一个神奇的值null
然后filter会跳过这些nulls
.
这意味着即使我已经知道要跳过此数据,也会调用过滤函数。
- 也许我应该使用另一个运算符
- 有没有办法在过滤器函数中使用可观察复选框
- 也许在组合函数中有一些方法可以表明我想跳过这个数据
你想要输出的图片不正确。在您的实现中,combineLatest
将 null
组合到流中:
interval 1--2--3--4--5--6--7--8--9
checkbox 0-----1--------0-----1---
combineLatest N--N--3--4--5--N--N--8--9
filter(NonNull)------3--4--5--------8--9
恕我直言,在 Rx Stream 中使用 null
作为信号并不好,开发人员很容易陷入 NullPointerException
.
要避免使用null
,有两个措施。第一个是将结果转换为 Pair,然后再应用 filter & map。
很简单Pair
Class:
public class Pair<T, U> {
T first;
U second;
public Pair(T first, U second) {
this.first = first;
this.second = second;
}
}
那么整个实现是这样的:
Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> new Pair<>(isChecked,intervalCounter)
).filter(pair -> pair.first)
.map(pair -> pair.second) // optional, do this if you only need the second part
数据流向:
interval 1 2 3 4
| | | |
checkbox F | T | F | T F
| | | | | | |
combineLatest F1 T1 T2 F2 F3 T3 F4
| | |
filter(first=T) T1 T2 T3
| | |
map(second) 1 2 3
或者,如果您可以在您的项目中 Java 8,请使用 Optional
来避免 null
,这与您的解决方案非常相似,但它让其他开发人员意识到流信号是 Optional
.
Observable<Optional<Integer>> o1 = Observable.combineLatest(
RxCompoundButton.checkedChanges(checkBox)
, Observable.interval(1, TimeUnit.SECONDS)
, (isChecked, intervalCounter) -> {
if (!isChecked) {
return Optional.empty();
} else {
return Optional.of(intervalCounter);
}
}
)
Observable<Integer> o2 = o1.filter(Optional::isPresent).map(Optional::get)
一些更简单的方法,与 combineLatest
不同,但与您想要的结果图片相同。
// Approach 1
Observable.interval(1, TimeUnit.SECONDS).filter( i -> checkBox.isEnabled())
// Approach 2
Observable.interval(1, TimeUnit.SECONDS)
.withLatestFrom(
RxCompoundButton.checkedChanges(checkBox),
(isChecked, intervalCounter) -> {
if (!isChecked) {
return Optional.empty();
} else {
return Optional.of(intervalCounter);
}
}).filter(Optional::isPresent).map(Optional::get)