​​​​​​​Rxjs map, mergeMap 和 switchMap 的区别与联系

2022-07-14,,,

前言

map、mergemap 和 switchmap 是 rxjs 中的三个主要运算符,在 sap spartacus 开发中有着广泛的使用场景。

map

map 是 observables 中最常见的运算符。 它的作用与数组中的映射相对相似。 map 接收从 observable 发出的每个值,对其执行操作并返回一个 observable(因此 observable 链可以继续)。

把它想象成一个函数,它将采用原始值和投影。 该函数将投影应用于所述值并在转换后返回它们。

让我们举个例子。 假设我们有一个 observable 数组。 这个数组是一个 person 的集合。 一个对象代表每个人,每个人都有自己的名字和喜欢的角色。 我们只对获取所有角色的列表感兴趣。

import { of } from 'rxjs';
import { map } from 'rxjs/operators';
const observable = of([
    {
        name: "parwinder",
        character: "calcifer"
    },
    {
        name: "laure",
        character: "alchemist"
    },
    {
        name: "eliu",
        character: "x-men"
    },
    {
        name: "robert",
        character: "link"
    }
]);
observable.pipe(
    map(arr => arr.map(person => person.character)) // loops over objects and returns characters
).subscribe(
    char => console.log(char) // ["calcifer", "alchemist", "x-men", "link"]
);

mergemap

mergemap 是 observable map 和 mege 的组合。 在实际项目中,经常需要 map 生成多个 observable。 例如,现在我有一个角色数组,对于每个角色,我想进行后端调用并获取一些信息。

看下面的例子:

import { of, from } from 'rxjs';
import { map } from 'rxjs/operators';
const dummyapi = (character) => { // fake api call function
  return of(`api response for character: ${character}`).pipe(
    delay(1000) // the fake api takes 1 second
  );
}
from(["calcifer", "alchemist", "x-men", "link"]) // characters i need to get information for
.pipe(
  map(arr => dummyapi(arr)) // generates 4 new observables
).subscribe( // subscribing observable (outer) of 4 observables (inner)
  data => data.subscribe(i => console.log(i)) // subscribing to inner observables
)

dummyapi 是现实项目中的典型例子:输入某个关键字,返回关键字对应的明细,包裹在一个 observable 对象里。也就是说,map 投影的输出是一个 observable,而不是普通对象,因此上面的代码编写了丑陋的嵌套 subscribe 来获取实际值。

使用 mergemap 后,这个操作符能够自动将 map 返回的 observable 进行 flatten 操作。使用 map 时丑陋的双重 subscribe 调用消失了。

import { of, from } from 'rxjs';
import { mergemap } from 'rxjs/operators';
const dummyapi = (character) => {
  return of(`api response for character: ${character}`)..pipe(
    delay(1000)
  );
}
from(["calcifer", "alchemist", "x-men", "link"])
.pipe(
  mergemap(arr => dummyapi(arr)) // gets 4 observable as api response and merges them
).subscribe( // we subscribe to one mapped and merged observable
  data => console.log(data)
)

switchmap

switchmap 的功能与 mergemap 的功能相同,但略有不同。 switchmap 将订阅外部 observable 中的所有内部 observable,但不会合并内部 observable。 它改为切换到最新的 observable 并将其传递给链。

它仍然提供一个 observable 作为输出,不是通过合并,而是通过仅从最新的 observable 发出结果的想法。

对于我们的最后一个示例,如果我们使用 switchmap,我们只会从最后一个 observable 中获取结果。

import { of, from } from 'rxjs';
import { switchmap, delay } from 'rxjs/operators';
const dummyapi = (character) => {
  return of(`api response for character: ${character}`).pipe(
    delay(1000)
  );
}
from(["calcifer", "alchemist", "x-men", "link"])
.pipe(
  switchmap(arr => dummyapi(arr))
).subscribe(
  data => console.log(data) // api response for character: link
)

有些场景是 switchmap 擅长的,比如所谓的 typehead.

想象这样一个场景:ui 上有一个输入框,我们在其中根据最终用户输入的内容,向其返回搜索结果。

如果用户打算输入 chase,开始输入 c,然后触发一个 api 调用。 然后客户继续输入 h,我们就必须再次针对 ch 调用一次后台 api。 此时,我们之前针对 c 的 api 调用已经毫无用处。 我们应该取消之前的 observable, 并订阅 ch 对应的 observable. 更一般性地说,我们需要切换到最新的 observable.

import { of, from } from 'rxjs';
import { switchmap, delay } from 'rxjs/operators';
const dummyapi = (character) => {
  return of(`search result for keyword: ${character}`).pipe(
    delay(1000)
  );
}
from(["c", "ch", "cha", "chas", "chase"]) // mimic key input in text field
.pipe(
  switchmap(arr => dummyapi(arr))
).subscribe(
  data => console.log(data) // search result for keyword: chase
)

到此这篇关于rxjs map, mergemap 和 switchmap 的区别与联系的文章就介绍到这了,更多相关rxjs map与mergemap 内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!

《​​​​​​​Rxjs map, mergeMap 和 switchMap 的区别与联系.doc》

下载本文的Word格式文档,以方便收藏与打印。