昨天咱们介绍了 Subject 是什么,今天要讲 Subject 一些应用方式,以及 Subject 的另外三种变形。javascript
昨天咱们讲到了 Subject 实际上就是 Observer Pattern 的实例,他会在内部管理一份 observer 的清单,并在接收到值时遍历这份清单并送出值,因此咱们能够这样用 Subject前端
var subject = new Rx.Subject();
var observerA = {
next: value => console.log('A next: ' + value),
error: error => console.log('A error: ' + error),
complete: () => console.log('A complete!')
}
var observerB = {
next: value => console.log('B next: ' + value),
error: error => console.log('B error: ' + error),
complete: () => console.log('B complete!')
}
subject.subscribe(observerA);
subject.subscribe(observerB);
subject.next(1);
// "A next: 1"
// "B next: 1"
subject.next(2);
// "A next: 2"
// "B next: 2"复制代码
JSBinjava
这里咱们能够直接用 subject 的 next 方法传送值,全部订阅的 observer 就会接收到,又由于 Subject 自己是 Observable,因此这样的使用方式很适合用在某些没法直接使用 Observable 的前端框架中,例如在 React 想对 DOM 的事件作监听前端框架
class MyButton extends React.Component {
constructor(props) {
super(props);
this.state = { count: 0 };
this.subject = new Rx.Subject();
this.subject
.mapTo(1)
.scan((origin, next) => origin + next)
.subscribe(x => {
this.setState({ count: x })
})
}
render() {
return <button onClick={event => this.subject.next(event)}>{this.state.count}</button>
}
}复制代码
从上面的代码能够看出来,由于 React 自己 API 的关系,若是咱们想要用 React 自订的事件,咱们没办法直接使用 Observable 的 creation operator 创建 observable,这时就能够靠 Subject 来作到这件事。ui
Subject 由于同时是 observer 和 observable,因此应用面很广除了前面所提的以外,还有上一篇文章讲的组播(multicase)特性也会在接下来的文章作更多应用的介绍,这里先让咱们来看看 Subject 的三个变形。this
不少时候咱们会但愿 Subject 能表明当下的状态,而不是单存的事件发送,也就是说若是今天有一个新的订阅,咱们但愿 Subject 能当即给出最新的值,而不是没有回应,例以下面这个例子spa
var subject = new Rx.Subject();
var observerA = {
next: value => console.log('A next: ' + value),
error: error => console.log('A error: ' + error),
complete: () => console.log('A complete!')
}
var observerB = {
next: value => console.log('B next: ' + value),
error: error => console.log('B error: ' + error),
complete: () => console.log('B complete!')
}
subject.subscribe(observerA);
subject.next(1);
// "A next: 1"
subject.next(2);
// "A next: 2"
subject.next(3);
// "A next: 3"
setTimeout(() => {
subject.subscribe(observerB); // 3 秒后才订阅,observerB 不会收到任何值。
},3000)复制代码
以上面这个例子来讲,observerB 订阅的以后,是不会有任何元素送给 observerB 的,由于在这以后没有执行任何 subject.next()
,但不少时候咱们会但愿 subject 可以表达当前的状态,在一订阅时就能收到最新的状态是什么,而不是订阅后要等到有变更才能接收到新的状态,以这个例子来讲,咱们但愿 observerB 订阅时就能当即收到 3
,但愿作到这样的效果就能够用 BehaviorSubject。.net
BehaviorSubject 跟 Subject 最大的不一样就是 BehaviorSubject 是用来呈现当前的值,而不是单纯的发送事件。BehaviorSubject 会记住最新一次发送的元素,并把该元素看成目前的值,在使用上 BehaviorSubject 建构式须要传入一个参数来表明起始的状态,范例以下code
var subject = new Rx.BehaviorSubject(0); // 0 为起始值
var observerA = {
next: value => console.log('A next: ' + value),
error: error => console.log('A error: ' + error),
complete: () => console.log('A complete!')
}
var observerB = {
next: value => console.log('B next: ' + value),
error: error => console.log('B error: ' + error),
complete: () => console.log('B complete!')
}
subject.subscribe(observerA);
// "A next: 0"
subject.next(1);
// "A next: 1"
subject.next(2);
// "A next: 2"
subject.next(3);
// "A next: 3"
setTimeout(() => {
subject.subscribe(observerB);
// "B next: 3"
},3000)复制代码
JSBin | JSFiddle
从上面这个范例能够看得出来 BehaviorSubject 在创建时就须要给定一个状态,并在以后任何一次订阅,就会先送出最新的状态。其实这种行为就是一种状态的表达而非单存的事件,就像是年龄跟生日同样,年龄是一种状态而生日就是事件;因此当咱们想要用一个 stream 来表达年龄时,就应该用 BehaviorSubject。
在某些时候咱们会但愿 Subject 表明事件,但又能在新订阅时从新发送最后的几个元素,这时咱们就能够用 ReplaySubject,范例以下
var subject = new Rx.ReplaySubject(2); // 重复发送最后 2 个元素
var observerA = {
next: value => console.log('A next: ' + value),
error: error => console.log('A error: ' + error),
complete: () => console.log('A complete!')
}
var observerB = {
next: value => console.log('B next: ' + value),
error: error => console.log('B error: ' + error),
complete: () => console.log('B complete!')
}
subject.subscribe(observerA);
subject.next(1);
// "A next: 1"
subject.next(2);
// "A next: 2"
subject.next(3);
// "A next: 3"
setTimeout(() => {
subject.subscribe(observerB);
// "B next: 2"
// "B next: 3"
},3000)复制代码
JSBin |
可能会有人觉得 ReplaySubject(1)
是否是就等同于 BehaviorSubject,实际上是不同的,BehaviorSubject 在创建时就会有起始值,好比 BehaviorSubject(0)
起始值就是 0
,BehaviorSubject 是表明着状态而 ReplaySubject 只是事件的重放而已。
AsyncSubject 是最怪的一个变形,他有点像是 operator last
,会在 subject 结束后送出最后一个值,范例以下
var subject = new Rx.AsyncSubject();
var observerA = {
next: value => console.log('A next: ' + value),
error: error => console.log('A error: ' + error),
complete: () => console.log('A complete!')
}
var observerB = {
next: value => console.log('B next: ' + value),
error: error => console.log('B error: ' + error),
complete: () => console.log('B complete!')
}
subject.subscribe(observerA);
subject.next(1);
subject.next(2);
subject.next(3);
subject.complete();
// "A next: 3"
// "A complete!"
setTimeout(() => {
subject.subscribe(observerB);
// "B next: 3"
// "B complete!"
},3000)复制代码
JSBin |
从上面的代码能够看出来,AsyncSubject 会在 subject 结束后才送出最后一个值,其实这个行为跟 Promise 很像,都是等到事情结束后送出一个值,但实务上咱们很是很是少用到 AsyncSubject,绝大部分的时候都是使用 BehaviorSubject 跟 ReplaySubject 或 Subject。
咱们把 AsyncSubject 放在大脑的深处就好
今天介绍了 Subject 的一些应用方式,以及 BehaviorSubject, ReplaySubject, AsyncSubject 三个变形各自的特性介绍,不知道读者么是否有收获呢? 若是有任何问题,欢迎在下方留言给我!