RxJS在看文档后彻底是一片懵,直到我在angular的应用中遇到不少须要实现的功能的时候...RxJS完美!html
Observable只是一个普通函数,要想让他有所做为,就须要跟observer一块儿使用;前者是受后者是攻。而这个observer(后面咱们会介绍)只是一个带有 next、error、complete 的简单对象而已。最后,还须要经过 subscribe 订阅来启动Observable;不然它是不会有任何反应;能够理解为陌*为了他们能在一块儿而提供的环境,而订阅也会返回一个可用于取消操做(在RxJS里叫 unsubscribe)。node
当Observable设置观察者后,而链接并获取原始数据的这个过程叫生产者,多是DOM中的 click 事件、input 事件、或者更加复杂的HTTP通讯。app
为了更好理解,先从一个简单的示例开始:函数
import { Component } from '@angular/core';
import { Observable, Subscription } from 'rxjs';
@Component({
selector: 'app-home',
template: `<input type="text"> `
})
export class HomeComponent {
ngOnInit() {
const node = document.querySelector('input[type=text]');
// 第二个参数 input 是事件名,对于input元素有一个 oninput 事件用于接受用户输入
const input$ = Observable.fromEvent(node, 'input');
input$.subscribe({
next: (event: any) => console.log(`You just typed ${event.target.value}!`),
error: (err) => console.log(`Oops... ${err}`),
complete: () => console.log(`Complete!`)
});
}
}
复制代码
示例中 Observable.fromEvent() 会返回一个Observable,而且监听 input 事件,当事件被触发后会发送一个 Event 给对应的observer观察者。ui
observer很是简单,像上面示例中 subscribe 订阅就是接收一个 observer 方法。this
通常在Angular咱们 subscribe 会这么写:spa
input$.subscribe((event: any) => {
});
复制代码
从语法角度来说和 subscribe({ next, error, complete })
是同样的。code
当Observable产生一个新值时,会通知 observer 的 next()
,而当捕获失败能够调用 error()
。component
当Observable被订阅后,除非调用observer的 complete()
或 unsubscribe()
取消订阅两状况之外;会一直将值传递给 observer。server
Observable的生产的值容许通过一序列格式化或操做,最终获得一个有价值的数据给观察者,而这一切是由一序列链式operator来完成的,每个operator都会产生一个新的Observable。而咱们也称这一序列过程为:流。
正如前面说到的,Observable能够链式写法,这意味着咱们能够这样:
Observable.fromEvent(node, 'input')
.map((event: any) => event.target.value)
.filter(value => value.length >= 2)
.subscribe(value => { console.log(value); });
复制代码
下面是整个顺序步骤:
假设用户输入:a Observable对触发 oninput 事件做出反应,将值以参数的形式传递给observer的 next()。 map() 根据 event.target.value 的内容返回一个新的 Observable,并调用 next() 传递给下一个observer。 filter() 若是值长度 >=2 的话,则返回一个新的 Observable,并调用 next() 传递给下一个observer。 最后,将结果传递给 subscribe 订阅块。 你只要记住每一次 operator 都会返回一个新的 Observable,无论 operator 有多少个,最终只有最后一个 Observable 会被订阅。
不要忘记取消订阅。
Observable 当有数据产生时才会推送给订阅者,因此它可能会无限次向订阅者推送数据。正由于如此,在Angular里面建立组件的时候务必要取消订阅操做,以免内存泄漏,要知道在SPA世界里懂得擦屁股是一件必须的事。
前面示例讲过,调用 subscribe() 后,会返回一个 Subscription 可用于取消操做 unsubscribe()。最合理的方式在 ngOnDestroy 调用它。
ngOnDestroy() {
this.inputSubscription.unsubscribe();
}
复制代码
一个很是优雅的取消订阅方法
Observable.fromEvent(node, 'input')
.takeUnit((event: any) => this.show)
.subscribe(value => { console.log(value); });
ngOnDestory(): void {
this.show = false;
}
复制代码
若是说 Observable 与 observer 是攻受结合体的话,那么 Subject 就是一我的即攻亦受。正由于如此,咱们在写一个Service
用于数据传递时,老是使用 new Subject。下面展现一个例子:
@Injectable()
export class MessageService {
private subject = new Subject<any>();
send(message: any) {
this.subject.next(message);
}
get(): Observable<any> {
return this.subject.asObservable();
}
}
复制代码
当F组件须要向M组件传递数据时,咱们能够在F组件中使用 send()。
constructor(myService: MyService)
ngOnInit() {
this.myService.send('message');
}
复制代码
M组件接收数据:
constructor(myService: MyService)
ngOnInit() {
this.myService.get.subscribe(data => {
console.log(data); // message
})
}
复制代码
在angular中EventEmitter至关因而Subject
// my-component.component.ts
@outPut() onChange = new EventEmitter<String>();
onClick() {
this.onChange.emit('hello world');
}
复制代码
<!-- parent-coponent.component.html -->
<my-component onChange="change($event)"></my-component>
复制代码
// parent-coponent.component.ts
change(value) {
console.log(value) // hello world
}
复制代码
RxJS有不少操做符,详情见个人另外一篇文章《RxJS(1) 常见Operators》。