动机
JavaScript的反应性扩展(RxJS)是_可观察_数据流的库。 RxJS在命令行执行ng new [name-of-application]
时安装Angular。这使用Angular命令行界面(CLI)。 RxJS补充了数据通过Observable
流程。 Observable
对象有助于_可迭代_数据的流动。
数据流不是主要用例。毕竟,数据流并行事件流。事件发出,因此应用程序知道数据何时到达。虽然事件流构成了RxJS补充的核心,但本文也将它们称为数据流。
流同步(立即)或异步(超时)执行。 RxJS通过Observable
数据流轻松处理这两种情况。严格的异步性是可以转换的。使用内存中的_可迭代_数据会立即发生,而外部数据获取需要时间。 Angular支持RxJS库,因此它可以处理带有数据流的两种用例。
反应式编程
在深入研究之前,了解支持RxJS库的范例非常重要。如上所述,它通过Observable
对象工作,该对象简化了事件发射数据。
RxJS围绕基本Observable
。它的整个图书馆补充了它的功能。 RxJS甚至包括其他对象,包括Subject
, Subscription
和Observer
。每个都是它自己的基本Observable
的自定义变体。
RxJS源于反应式编程范例。这种范式引入了_可观察的_模式。在其中存在这个关键思想:单个Observable
在其所有Observer
被通知时发出。 Observer
订阅 Observable
以便他们收到通知。此通知可能意味着几件事。
它可能表示数据更改或数据到达,如本文中所述。它可能表示影响Observer
的应用程序的某些部分发生了变化。
这种_可观察的_模式也力图解耦概念。 Observable
应该能够在没有任何Observer
的情况下运行,反之亦然。这通常意味着它们可以独立而不是完全相互作用。
观测
为了快速重申,可以观察到 Observable
。这样一个Observable
可以根据数据流为其依赖项提供反馈。在RxJS中, Observable
是它自己的工厂函数,用于创建Observable
对象。他们的基本蓝图如下。
import { Observable } from 'rxjs';
const observable = Observable.create((source) => {
source.next(data);
});
.next
在向观察者发出事件的同时传递数据。 Observable
使用.next
从其.create
回调中发出数据。它接受一个表示要发出的数据的参数。 Observable
还没有在JavaScript中实现。 RxJS提供了其库的替代品。
下一步是观察员。要告诉函数或对象观察 Observable
,使用以下语法: observable.subscribe(observer)
。另一种看待它的方法是producer.subscribe(consumer)
。 Observable通过调用.next
生成数据。然后在接收数据时通知消费者。
import { Observable } from 'rxjs';
const observable = Observable.create((source) => {
source.next("Hello");
source.next("World!");
});
observable.subscribe((word) => console.log(word));
// console output
/*
Hello
World!
*/
两个.next
调用发生在Observable
的.create
回调(数据生成器)中。这导致观察者(数据使用者)的两个单独的控制台输出。
.next
的两次调用表示同步数据流。 Streams将数据概念化为线性有序流。它可以根据数据的可用性同步或异步解析。
如果包含流的数据容易获得,则它同步执行。否则,流将超时异步解析。在任何一种情况下,数据的顺序总是相同的,这取决于observable中.next
的调用。
Observable
像队列一样运行。在一段数据上调用.next
会将其推送到队列的后面。一旦解决了数据从前面弹出。
Observable
数据流具有巨大的吸引力。它们在顺序上是确定性的,并且根据数据可用性而合理地执行。此外,任何数量的观察者都可以_观察_数据源Observable
。这意味着数据可以生成一次并在一次操作中随处发出。
回调函数不是使用数据的唯一方法。观察者可以作为生产者和消费者彼此链接。
const observableI = Observable.create((source) => {
source.next("Hello World!");
});
const observableII = new Observable().subscribe((v) => console.log(v));
observableI.subscribe(observableII);
// console output
/*
Hello World!
*/
.subscribe
位于Observable
对象上。你可以用一个Observable
作为它的源(生产者)和另一个observable作为它的参数(消费者)来调用它。数据可以通过任意数量的可观察量流动(发射)。
JavaScript的反应性扩展(RxJS)
流数据很好,但是如果observable不能编辑流,那又有什么意义呢?这就是RxJS库发挥作用的地方。它提供了对数据流执行各种突变的运算符。
Angular利用这些运算符来转换传入的数据。开发人员可以使用RxJS运算符从传入流中删除任何不必要的数据。这样可以节省内存并减少对额外转换逻辑的需求。
RxJS提供与标准Observable
偏差,如Subject
, Subscription
和Observer
。将这些偏差视为传统Observable
特殊风味。他们没有必要使用图书馆。也就是说,像Subject
这样的变体具有超越标准Observable
令人难以置信的用例。
本文坚持使用标准的Observable
。来自RxJS的所有数据流运算符都通过Observable
工作。
许多核心RxJS运营商来自JavaScript的Array Extras。 Array对象的原型包含许多与RxJS库相似的内容。这些也被称为'额外'。数组是类似于可观察数据流的类似流的结构。
为了更好地理解RxJS运算符,本文将简要介绍JavaScript的Array Extras。每个功能几乎与其RxJS对应功能相同。区别仅在于数据的格式(可迭代数组与可迭代流)。
阵列附加功能
数组包含许多实用方法。这些方法称为Array Extras。它们都存在于Array对象的原型中。下面的列表包含五个与RxJS并行的额外内容。
.reduce
.filter
.map
.every
.forEach
对于每个示例,数组迭代自身以产生最终结果。
.reduce
将数组最小化为单个值。 .filter
使用布尔值评估修剪数组。 .map
逐个元素地转换数组。 .every
根据布尔条件计算整个数组的true或false。 .forEach
遍历数组的元素。
数组模型流。它们是彼此按顺序并逐个迭代。 Observable以类似的方式将数据元素简化为他们的观察者。这就是RxJS在其库中包含每个Array Extra的逻辑对应的原因。当然,与Array Extras相比,RxJS提供了更多自己的运算符。
基本的RxJS运算符
实际上有一整套图书馆的RxJS运营商。本文重点介绍下面列出的Array Extras的启发。
.reduce
.filter
.map
.every
.forEach
之前的列表没有任何变化。您对Array Extras的理解适用于RxJS运算符。唯一能看到的是一个名为.pipe
的函数,在接下来的几个例子中会有很多用处。 .pipe
链接RxJS运算符。来自前一个运算符的结果将进入下一个运算符直到最终运算符。然后,结果数据从可观察流中发出。
请注意下面的标准示例。使用它来比较每个运营商对发出的数据流的影响。
import { Observable, from } from 'rxjs';
const stream: number[] = [1, 2, 3, 4, 5];
const observable: Observable<number> = from(stream);
observable.subscribe((val: number) => console.log(val));
// console output
/*
1
2
3
4
5
*/
.from
将数组转换为Observable
对象,该对象在每个数组元素上调用.next
。 .pipe
函数接受任意数量的参数作为数组运算符。这是每个运营商实施的地方。运算符按照它们的实现顺序执行简化数据,作为.pipe
参数。
减少
.reduce
在发出之前将数据流最小化为单个值。
import { reduce } from 'rxjs/operators';
const stream: number[] = [1, 2, 3, 4, 5];
const observable: Observable<number> = from(stream).pipe(
reduce((accum, val) => (accum + val))
);
observable.subscribe((val: number) => console.log(val));
// console output
/*
15
*/
过滤
.filter
修剪流,消除不满足其条件的流值。
import { filter } from 'rxjs/operators';
const stream: number[] = [1, 2, 3, 4, 5];
const observable: Observable<number> = from(stream).pipe(
filter((val) => (val % 2 === 0)) // filters out odd numbers
);
observable.subscribe((val: number) => console.log(val));
// console output
/*
2
4
*/
地图
.map
定位并转换每个正在进行的流值。
const stream: number[] = [1, 2, 3, 4, 5];
const observable: Observable<number> = from(stream).pipe(
map((val) => (val + 1))
);
observable.subscribe((val: number) => console.log(val));
// console output
/*
2
3
4
5
6
*/
挑战:每一个和每一个
了解.every
和.forEach
作为Array Extras,尝试将它们实现为RxJS运算符。请随意使用前面的示例进行指导。经过大量阅读后,一点点的练习都有很长的路要走!
Angular中的HTTP
本节将RxJS和Angular结合在一起,展示它们如何相互作用。通常,Angular提供的服务将提供Observable
。然后, Observable
的数据流可以使用带有.pipe
RxJS运算符进行.pipe
。
Angular通过@angular/common/http
提供HttpClient
服务。 HttpClientModule
也来自@angular/common/http
并导出HttpClient
服务。应用程序的根模块必须导入HttpClientModule
。这使得HttpClientModule
从应用程序的任何位置注入 。
HttpClientModule
服务发出HTTP请求。这些请求是异步的。让他们对Angular感兴趣的是如何处理请求。每个请求都会返回一个Observable
。 RxJS可以把它带走。
即将到来的示例使用由Typicode构建的公共API。 API为每个异步GET
请求提供100个元素的数组。
// ./models/post.model.ts
export interface Post {
userId: number;
id: number;
title: string;
body: string;
}
// ./services/json.service.ts
import { HttpClient } from '@angular/common/http';
import { Injectable } from '@angular/core';
import { Observable, from } from 'rxjs';
import { switchMap, map, filter, reduce } from 'rxjs/operators';
import { Post } from '../models/post.model';
@Injectable({
providedIn: 'root'
})
export class JsonService {
constructor(private http: HttpClient) { }
getPostsByUserId(id: number): Observable<any> {
const trim$ = (stream) => from(stream)
.pipe(
filter((post: Post) => +post.userId === +id),
map((post: Post) => ({ title: post.title, body: post.body })),
reduce((accum: Post[], post: Post) => accum.concat([post]), [])
);
return this.http.get("https://jsonplaceholder.typicode.com/posts")
.pipe(
switchMap((value) => trim$(value))
);
}
}
// ./components/example/example.component.ts
import { Component } from '@angular/core';
import { JsonService } from '../../services/json.service';
import { Post } from '../../models/post.model';
@Component({
selector: 'app-example',
template: `
<h1>Request User Posts</h1>
<span>User: </span><input #userInput>
<button (click)="requestForPosts(userInput.value)">REQUEST</button>
<hr>
<ul>
<div *ngIf="userPosts">
<div *ngFor="let post of userPosts">
<h3>{{ post.title }}</h3>
<p>{{ post.body }}</p>
</div>
</div>
<h3 *ngIf="!userPosts">No posts shown...</h3>
</ul>
`
})
export class ExampleComponent {
userPosts: Post[];
constructor(private json: JsonService) { }
requestForPosts(id: number): void {
this.json.getPostsByUserId(id)
.subscribe((posts: Post[]) => { this.userPosts = posts.length > 0 ? posts : null; });
}
}
json.service.ts
创建一个Observable
的代表component.example.ts
。组件可以订阅返回的Observable
。到Observable
解析数据流时,只会发出一个值。
对jsonplaceholder.typicode.com
的请求产生一个包含100个帖子的单个数组。通过HttpClient
的请求产生一个Observable
。运算符switchMap
返回另一个Observable
,它覆盖当前流。变量trim$
将Observable
存储为其值。将$
附加到用于存储Observable
的变量是惯例。
from
jsonplaceholder.typicode.com
将数组转换为100值发射的Observable
。然后,RxJS运算符筛选流中的每个数据。他们删除与请求无关的流值。进行数据修剪,以使流值保持不必要的信息。最终结果再次作为单个数组连接在一起,向其观察者发出一个数组。
在component.example.ts
,JsonService引用返回刚才描述的Observable
。此方法在组件模板中单击按钮时调用。模板中的输入框也提供单个id
参数。
按下按钮,JsonService返回一个发出单个数组的Observable
。 .subscribe
对返回的Observable
调用。然后,该组件将userPosts
的值设置为等于发出的数组。
角度变化检测可以获取类数据的变化。模板更新和*ngFor
确保*ngFor
每个数组元素userPosts
呈现自己的模板元素。
结论
RxJS提供核心Observable
及其运算符。该库使用ng new [name-of-app]
(Angular CLI)从命令行自动安装。 RxJS核心类型和运算符分别下载到rxjs
和rxjs/operators
。
即使您不使用CLI, HttpClient
等服务仍然可用。如果RxJS不可用,则服务返回Promise
而不是Observable
。与Observable
不同, Promise
对象是JavaScript的原生对象。这可能会在下一个官方JavaScript版本中发生变化。
也就是说,充分利用RxJS!任何可迭代的结构都可以容纳Observable
。有了它,整个RxJS库变得可用。其运营商有效地将数据从流转换为结果。此外,观察者可以订阅结果,从而提高数据的整体可移植性。
更多Angular教程
学习更多Angular教程