有人可以解释这三种变体之间的区别吗?
http://jsfiddle.net/8vx2g3fr/2/
请问您能帮助我了解问题是什么以及如何使第三个变体处理所有事件?
let bs = new Rx.Subject();
bs
.subscribe(v=>{
console.log("in", v);
if (v % 2 == 0) {
setTimeout(()=>{
console.log(" out", v, "->" , v + 1);
bs.next(v+1);
}, 0);
}
});
bs.next(0);
bs.next(2);
输出:
in 0
in 2
out 0 -> 1
in 1
out 2 -> 3
in 3
let bs2 = new Rx.Subject();
bs2
.subscribe(v=>{
console.log("in", v);
if (v % 2 == 0) {
Rx.Observable.interval(0).take(1)
.map(()=>{console.log(" out", v, "->" , v + 1);return v+1;})
.subscribe(bs2);
}
});
bs2.next(0);
bs2.next(2);
输出:
in 0
in 2
out 0 -> 1
in 1
out 2 -> 3
let bs3 = new Rx.Subject();
bs3
.switchMap(v=>{
console.log("in", v);
if (v % 2 == 0) {
return Rx.Observable.interval(0).take(1)
.map(()=>{console.log(" out", v, "->" , v + 1);return v+1;});
}
return Rx.Observable.empty();
}).subscribe(bs3);
bs3.next(0);
bs3.next(2);
输出:
in 0
in 2
out 2 -> 3
in 3
答案 0 :(得分:4)
这实际上是预期的行为。
令人困惑的是,当您多次重复使用Subject
和take()
等运算符时会发生这种情况。
运营商take(1)
仅获取一个值并发送complete
通知。由于Subject
,.subscribe(bs2)
会收到此通知。现在是最重要的部分
当Subject
收到complete
或error
通知时,它会将自己标记为已停止。这意味着永远不会汇出任何项目或通知,这是Rx中的正确和预期行为。通知complete
或error
必须是最后一次排放。
所以Subject
由第一个take(1)
完成,由值0
(bs2.next(0)
调用)触发。
然后当值2
触发Observable.interval(0).take(1)
收到的Subject
的第二次运行时,它会被自动忽略,因为Subject
已标记为已停止。
第三个演示中的过程完全相同。
您可以在Subject.ts
中的源代码中看到它: