前言

我们一样从简单和常用的入手.

第一篇介绍了 Creation Operators

上一篇介绍了 Filter Operators

这一篇来到 Join Creation Operators.

参考

Docs – Join Creation Operators

merge

merge 会同时 subscribe 所有的 Observable. 任何一个发布都会接收.

const s1 = new Subject();
const s2 = new Subject();
merge(s1, s2).subscribe(v => console.log(v)); // s1..s2
s1.next('s1');
s2.next('s2');

merge 会创建出一个新的 Observable. 当被订阅时, 它会去 subscribe s1 和 s2.

一旦 s1 或 s2 发布, merge observable 也会随之发布. merge 的 subscriber 就会接收.

merge 常用来实现多种不同监听, 但要执行相同逻辑的场景.

比如 click submit button 或 keydown enter 都会执行 ajax search. 那么我们就可以写 merge(click$, enter$).subscrube(() => ajax());

combineLatest

combineLatest 和 merge 类似, 创建新的 Observable 并且订阅所有 Observable

const s1 = new Subject();
const s2 = new Subject();
combineLatest([s1, s2]).subscribe(([v1, v2]) => console.log([v1, v2])); // [s1, s2]
s1.next('s1'); // 不会接收, 因为 s2 还没有值
s2.next('s2'); // 接收, 因为 s1, s2 都有值了

但它和 merge 有 2 个区别

1. merge 每次只接收到 1 个值 (最新发布的那个), 而 combineLatest 会接收到所有 Observable 的值 (它是一个 Array, 因为每一次发布值都会被 cache 起来)

2. combineLatest 必须等到所有的 Observable 最少发布一次以后, 才会开始接收.

上面的例子中, s1.next('s1') 的时候, console 没有响, 因为 s2 这时还没有任何发布记入. 只有当 s2.next 以后, s1, s2 都有了发布记入, console 才响, 同时获取到了 s1, s2 的最新的值.

combineLatest 在开发中也是经常需要使用到的, 尤其是当我们要 merge 的效果, 同时希望它返回当前所有 Observable 最新值的时候.

以前遇过的坑

const s1 = new Subject();
combineLatest([s1, s1]).subscribe(v => console.log(v));
s1.next('value'); // [value, value]
s1.next('value2'); // [value2, value]..[value2, value2]

效果

虽然都是 s1, 但 combineLatest 会把它们当作不同的 Observable, 会有 2 个订阅, 而 s1.next 就想等于 2 个 Obserable 都发布了值, 但依然是有顺序的哦, 一前一后.

所以会出现 3 个 console. 第二个 console 的接收值是 ['value2', 'value'] 因为第一个 Observable 发布, 第二个用的是 cached value.

zip

zip 和 combineLatest 有点像, 它们都会接收到所有 Observable 最新的值. 只是接收的时机不一样.

多个 Observable 发布的次数往往是没有规律的. 可能 o1 发布了 5 次, o2 才 2 次.

combineLatest 它不管大家的次数, 只要任何一个 Observable 发布, 那就把其它所有的值拉出来, 一起发布出去.

zip 则讲究次数, 只有当每一个 Observable 都达到相同次数, 它才会把每一个 Observable 那一次数的值取出来, 一起发布出去.

看下面这个例子

  const s1 = new Subject();
const s2 = new Subject();
combineLatest([s1, s2]).subscribe(v => console.log(v));
s1.next('s1');
s2.next('s2'); // [s1, s2]
s1.next('s11'); // [s11, s2]
s1.next('s12'); // [s12, s2]
s1.next('s13'); // [s13, s2]
s2.next('s21'); // [s13, s21] zip([s1, s2]).subscribe(v => console.log(v));
s1.next('s1');
s2.next('s2'); // [s1, s2]
s1.next('s11');
s1.next('s12');
s1.next('s13');
s2.next('s21'); // [s11, s21]

zip 最终 console 只响了 2 次. 而 combineLatest 响了 5 次.

虽然 s1 发布了 5 次, 但 zip 要求所有 Observable 必须要有相同次数, 而 s2 只有 2 次, 所以最终只能发布 2 次而已. 而第二次的发布, s1 的 value 并不是它最新的 value 而是它第 2 次发布的 value 哦.

concat

concat 和 merge 类似, 它会 subscribe 所有 Observable 但它不像 merge 那样一开始就全部 subscribe. 它会挨个挨个去 subscribe. 当第一个 Observable complete 以后才去 subscribe 第二个.

const s1 = new Subject();
const s2 = new Subject();
concat(s1, s2).subscribe(v => console.log(v));
s1.next('s1'); // 接收 s1
s2.next('s2'); // 没接收, 因为 s1 还没有 complete, s2 还没有开始 subscribe
s1.complete();
s2.next('s2'); // 接收 s2

forkJoin

forkJoin 好比 Promise.all, 当所有的 Observable complete 后, 它才会接收, 并且得到所有 Observable 最终的值.

const s1 = new Subject();
const s2 = new Subject();
forkJoin([s1, s2]).subscribe(v => console.log(v)); // [s1, s2]
s1.next('s1');
s2.next('s2');
s1.complete();
s2.complete(); // 这时才开始接收

race

race 就是比赛, 第一个发布的 Observable 将保留订阅, 其余的就淘汰退订

const s1 = new Subject();
const s2 = new Subject();
race([s1, s2]).subscribe(v => console.log(v)); // s2..s21
s2.next('s2'); // 接收
s1.next('s1'); // 不接收, 因为 s2 胜出, s1 已经被退订了
s2.next('s21'); // 接收 只有胜出的 s2 发布才会被接收

partition

partition 的作用是把 1 个 Observable 拆分成 2 个.

上一篇介绍 filter operator 时, 有写过下面这个 odd, even 例子

const source = from([1, 2, 3, 4]);
const odd$ = source.pipe(filter(v => v % 2 !== 0));
const even$ = source.pipe(filter(v => v % 2 === 0));
odd$.subscribe(v => console.log(v)); // 1..3
even$.subscribe(v => console.log(v)); // 2..4

它可以用 partition 改写成

const [odd$, even$] = partition(source, v => v % 2 !== 0);

条件 true 表示它属于第一个 Observable, false 则进入第二个 Observable.

注: source 没有 share 的概念哦,odd$ even$ 是两个独立的 stream。

一句话总结

merge : 任何一个发布, 接收最新发布值

combineLatest : 任何一个发布, 接收所有最新的值 (开始接收条件: 所有 Observable 最少要有一次发布)

zip : 和 combineLatest 一样, 但它 care Observable 发布的次数. 当所有 Observable 都满足 n 次的时候发布, 接收第 n 次所有的值.

concat : 上一个 Observable complete 了才 subscribe 下一个

forkJoin: Promise.all, 所有 Observable complete 后, 接收所有最终的值

race : 第一个发布的 Observable 继续订阅, 退订其余的 Observale

partition : 通过 if else 把一个 Observable 拆分成 2 个

RxJS 系列 – Join Creation Operators的更多相关文章

  1. RxSwift 系列(三) -- Combination Operators

    RxSwift 系列(三) -- Combination Operators 前言 本篇文章将要学习如何将多个Observables组合成一个Observable. Combination Opera ...

  2. [RxJS] Creation operators: interval and timer

    It is quite common to need an Observable that ticks periodically, for instance every second or every ...

  3. [RxJS] Creation operators: empty, never, throw

    This lesson introduces operators empty(), never(), and throw(), which despite being plain and void o ...

  4. [RxJS] Creation operators: fromEventPattern, fromEvent

    Besides converting arrays and promises to Observables, we can also convert other structures to Obser ...

  5. [RxJS] Creation operators: from, fromArray, fromPromise

    The of() operator essentially converted a list of arguments to an Observable. Since arrays are often ...

  6. Apache Flink 漫谈系列 - JOIN 算子

    聊什么 在<Apache Flink 漫谈系列 - SQL概览>中我们介绍了JOIN算子的语义和基本的使用方式,介绍过程中大家发现Apache Flink在语法语义上是遵循ANSI-SQL ...

  7. RxSwift 系列(四) -- Transforming Operators

    前言 本篇文章将要学习RxSwift中四种转换操作符: map flatMap flatMapLatest scan map 通过使用一个闭包函数将原来的Observable序列转换为一个新的Obse ...

  8. RxSwift 系列(七) -- Connectable Operators

    前言 本篇文章将要学习RxSwift中连接操作符. Connectable Observable在订阅时不发射事件消息,而是仅当调用它们的connect()方法时才发射消息,这样就可以等待所有我们想要 ...

  9. rxjs笔记(未完成)

    首先是 Observable 和promise的区别, 1返回值个数,Observable 可以返回0到无数个值. 2.Promise主动推送,控制着"值"何时被 "推送 ...

  10. RxJS——Operators

    RxJS 的操作符(operators)是最有用的,尽管 Observable 是最基本的.操作符最基本的部分(pieces)就是以申明的方式允许复杂的异步代码组合简化. 什么是操作符? 操作符是函数 ...

随机推荐

  1. .NET6 API 部署标准流程

    一.安装dotnet环境 #第一步:将 Microsoft 包签名密钥添加到受信任密钥列表,并添加 Microsoft 包存储库 sudo rpm -Uvh https://packages.micr ...

  2. oeasy教您玩转vim - 12 - # 词头词尾

    词头词尾 回忆上节课内容 我们这次学了向前一个单词 w 意思是 word 还学习了向后一个单词 b 意思是 backward 这俩命令都落在单词的第一个字母 还有什么好玩的命令吗? 动手练习 我们可以 ...

  3. [oeasy]python0026_刷新时间_延迟时间_time_sleep_死循环_while_True

    ​ 刷新时间 回忆上次内容 time 是一个 ​​module​ import 他可以做和时间相关的事情 time.time() 得到当前时间戳 time.localtime() 得到本地时间元组 l ...

  4. 解读GaussDB(for MySQL)灵活多维的二级分区表策略

    本文分享自华为云社区<GaussDB(for MySQL)创新特性:灵活多维的二级分区表策略>,作者:GaussDB 数据库. 背景介绍 分区表及二级分区表的功能,可以让数据库更加有效地管 ...

  5. CF1359A 题解

    洛谷链接&CF 链接 题目简述 共有 \(T\) 组数据. 对于每组数据给出 \(n,m,k\),表示 \(k\) 名玩家打牌,共 \(n\) 张牌,\(m\) 张王,保证 \(k \mid ...

  6. SEO自动外链蜘蛛池工具促进百度快速收录怎么样 跟大家详谈一下

    此工具集成市面上所有自动外链网站的资源链接,经过合并.去重.筛选.验证 总结出最终的外链资源 ,软件实时更新 本软件将您繁杂的外链推广转为自动化进行,并且加入站群的支持,您只需要将你的站群域名粘贴到软 ...

  7. 【Vue】Re02 指令:第一部分

    一.v-once指令 用于固定一次性赋值,后续Vue实例的赋值更改将不再对v-once指令的元素有效 <!DOCTYPE html> <html lang="en" ...

  8. 深度学习框架Theano停止维护

    Theano停止开发的声明地址: https://groups.google.com/g/theano-users/c/7Poq8BZutbY/m/rNCIfvAEAwAJ 原文内容: Dear us ...

  9. conda环境下安装nvidia-nvcc

    参考: https://www.cnblogs.com/littletreee/p/17234053.html conda安装Pytorch或TensorFlow的时候是默认不安装nvcc,但是有时候 ...

  10. 美国空军未来利用AI控制无人僚机执行空战或对地作战任务成为可能——AI算法强化学习控制无人机执行空战或对地作战成为可能

    来源: https://export.shobserver.com/baijiahao/html/640202.html https://baijiahao.baidu.com/s?id=177346 ...