目录

  • 前言
  • 观察者模式
  • 迭代器模式
  • RxJS 中两种模式的结合和实现
  • 小结
  • 参考

1. 前言

RxJS 是一个库,它通过使用observable(可观察对象)序列来编写异步和基于事件的程序。其结合了观察者模式迭代器模式使用集合的函数式编程,以一种理想方式来管理事件序列所需要的一切。

本文将主要探讨观察者模式、迭代器模式以及它们如何在 RxJS 中被应用。

2. 观察者模式

实现了生产者(事件的创建者)和消费者(事件的监听者)的逻辑分离关系。

浏览器 DOM 事件的监听和触发应该是 Web 前端最典型的观察者模式的实现。

document.body.addEventListener('click', function listener(e) {
console.log(e);
}); document.body.click(); // 模拟用户点击

监听:通过addEventListenerdocument.body节点绑定一个click事件的事件处理函数。

触发:当用户点击页面(body)时,body节点将会触发绑定的事件处理函数。

关系图如下:

3. 迭代器模式

可以让用户通过特定的接口访问集合中的每一个元素而不用了解底层的实现。

从 ES 6 开始,引入的一种新的遍历机制——迭代器,其就是迭代器模式在 JavaScript 中的一种实现。在 JavaScript 中,迭代器是一个对象,它定义一个序列,并在终止时可能返回一个返回值。 更具体地说,迭代器是通过使用 next() 方法实现 Iterator protocol (迭代器协议)的任何一个对象,该方法返回具有两个属性的对象: valuedone ,其中value代表具体返回值,done表示是否已经迭代完毕。

StringArrayMapSet 等都是内置可迭代对象,它们的原型对象都拥有一个 Symbol.iterator 方法。

const arr = ['a', 'b'];
const iterator = arr[Symbol.iterator](); // 获取迭代器对象 iterator.next(); // { value: 'a', done: false }
iterator.next(); // { value: 'b', done: false }
iterator.next(); // { value: undefined, done: true }

我们常常用for-of 循环来遍历可迭代对象:

const arr = ['a', 'b'];

for (let value of arr) {
console.log(value); // a b
}

for-of语法是为了方便遍历可迭代对象,其内部实现调用的是Symbol.iterator方法,类似下面的代码:

const arr = ['a', 'b'];
const iterator = arr[Symbol.iterator](); let result = iterator.next();
while (!result.done) {
console.log(result.value); // a b
result = iterator.next();
}

迭代器的特点:

  1. 访问集合中的内容而不用了解底层的实现。
  2. 提供了一个统一的接口遍历不同的集合结构,从而支持同样的算法在不同的集合结构上进行操作。

4. RxJS 中两种模式的结合和实现

RxJS 中包含两个基本概念:**Observable **和 Observer

Observable 作为可观察对象(被观察者),是一个可调用的未来值或事件的集合(异步或同步数据流)。

Observer 作为观察者,是一个回调函数的集合,它知道如何去监听由Observable提供的值。

ObservableObserver之间的订阅发布关系(观察者模式)如下:

订阅:Observer 通过 Observable 提供的 subscribe() 方法订阅 Observable

发布:Observable 通过 Observer 提供的 next 方法向 Observer 发布事件。

两者关系的伪代码如下:

// Observer
const observer = {
next(value) {
console.log(value);
}
}; // Observable
function Observable (observer) {
setTimeout(()=>{
observer.next('A');
}, 1000);
} // subscribe
Observable(observer);

从上可知,所谓订阅,就是将观察者Observer注入到可观察对象Observable中。

在 RxJS 中,Observer 除了有 next 方法来接收 Observable 的事件外,还提供了另外的两个方法:error()complete(),来处理异常和完成状态。

const observer = {
next(value) { /* 处理值 */ },
error(err) { /* 处理异常 */ },
complete() { /* 处理已完成态 */ }
};

结合迭代器 Iterator 来理解Observer的三个方法:

  • next()Observer通过提供 next 方法来接受Observable流(集合),是一种 push 形式(推送)。

    • 对比 Iterator,则是通过调用 iterator.next() 拿值,是一种 pull 的形式(拉取)。
  • complete():当不再有新的值发出时,将触发 Observercomplete 方法。
    • 对比 Iterator ,则是在 next() 的返回结果中的 donetrue 时,则表示 complete
  • error():当处理事件中出现异常时,通过try-catch捕获异常,Observer 提供 error 方法来接收错误进行统一处理。

一个简单的 RxJS 订阅-发布实例:

import { Observable } from 'rxjs';

const observable = new Observable(function (observer) {
// 通知观察者
observer.next('a');
observer.next('b');
observer.complete(); // 将取消该观察者的订阅 // observer.error(new Error('err')); observer.next('c'); // 由于已经 complete,所以不会再发送
}); // 定义观察者,next、complete、error 方法处理流的不同状态
const observer = {
next: (value) => console.log(value),
error: err => console.error('Observer got an error: ' + err),
complete: () => console.log('Observer got a complete notification')
} // 订阅 Observable 并执行
const subscription = observable.subscribe(observer); // 将返回一个可取消的订阅对象 subscription

执行结果:

a
b

5. 小结

一句话概述 RxJS 中实现的观察者+迭代器模式:就是将观察者Observer注入到可观察对象Observable中,然后在可观察对象Observable中通过调用Observer提供的 nextcompleteerror 方法处理流的不同状态,以实现对数据流的一种顺序访问处理。

6. 参考

RxJS 中文文档

Rx.js实现原理浅析

RxJS 中的观察者和迭代器模式的更多相关文章

  1. 迭代器模式的一种应用场景以及C#对于迭代器的内置支持

    迭代器模式 先放上gof中对于迭代器模式的介绍镇楼 意图 提供一种方法顺序访问一个聚合对象中各个元素, 而又不需暴露该对象的内部表示. 别名 游标(Cursor). 动机 一个聚合对象, 如列表(li ...

  2. (十七)迭代器模式详解(foreach的精髓)

    作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 各位好,很久没以LZ的身份和 ...

  3. 设计模式之迭代器模式详解(foreach的精髓)

    作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 各位好,很久没以LZ的身份和 ...

  4. js设计模式(四)---迭代器模式

    定义: 迭代器模式是指提供一种方法,顺序访问一个聚合对象中的各个元素,而又不需要暴露该对象的内部表示,迭代器模式可以把迭代的过程从业务逻辑中分离出来,使用迭代器模式,即使不关心对象的内部构造,也可以按 ...

  5. Java设计模式学习记录-迭代器模式

    前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...

  6. java设计模式之迭代器模式

    一.迭代器模式简介 迭代器模式提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部的表示. 迭代器模式让我们能游走于聚合内的每一个元素,而又不暴露内部的表示.把游走的任务放在迭代器上,而不是 ...

  7. 十七:迭代器模式详解(foreach的精髓)

    定义:提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示. 从定义中可以看出,迭代器模式是为了在不暴露该对象内部表示的情况下,提供一种顺序访问聚合对象中元素的方法.这种思想在JA ...

  8. 行为型模式(三) 迭代器模式(Iterator)

    一.动机(Motivate) 在软件构建过程中,集合对象内部结构常常变化各异.但对于这些集合对象,我们希望在不暴露其内部结构的同时,可以让外部客户代码透明地访问其中包含的元素:同时这种"透明 ...

  9. Java迭代器模式

    迭代器模式是Java和.Net编程环境中非常常用的设计模式.此模式用于以顺序方式访问集合对象的元素,而不需要知道其底层表示. 迭代器模式属于行为模式类别. 实现实例 在这个实例中,将创建一个Itera ...

随机推荐

  1. python--动态网页渲染pyqt5

    原文:https://blog.csdn.net/tymatlab/article/details/78647543 PyQt5 渲染动态网页 示例代码: # -*- coding: UTF-8 -* ...

  2. linux专题(八):用户组管理

    http://dwz.date/UDf 每个用户都有一个用户组,系统可以对一个用户组中的所有用户进行集中管理.不同Linux 系统对用户组的规定有所不同,如Linux下的用户属于与它同名的用户组,这个 ...

  3. Odoo13之在tree视图左上角添加自定义按钮

    前言 首先展示效果图,如下图所示,在资产设备模块tree视图的左上角添加了一个同步资产的按钮. 要完成按钮的添加,分为四步,分别是: 1.编写xml文件,找到相关模型tree视图,并给模型tree视图 ...

  4. Babel:下一代Javascript语法编译器

    定义 Babel是一个Javascript的编译器,通过它你可以将一些新版本的ECMAScript语法转换成低版本的语法.以便能够在低版本的浏览器或者其它环境平稳运行. 截至目前笔者写这篇文章的时候, ...

  5. 年薪30W+高薪测试技术要掌握哪些?

    职业技能一 1. 软件测试: 1) 熟练灵活地运用等价类.边界值.判定表法.因果图法等各种方法设计测试用例,包括单元测试.集成测试.系统测试用例设计. 2) 牢固掌握了软件测试计划.测试日报.测试报告 ...

  6. SQLite数据库多平台应用及常见错误分析

    SQLite是一个软件库,实现了自给自足的.无服务器的.零配置的.事务性的SQL数据库引擎.SQLite是世界上最广泛部署的数据库引擎之一.SQLite源代码开放,没有授权限制.正是因为其免费.轻巧. ...

  7. PyQt5绘图

    QPainter 功能:QPainter实现在QWidget上画图功能 说明:绘图必须在paintEvent中完成,且要在bengin和end之间作图 接口: 方法 描述 begin 开始画图 end ...

  8. Azure Traffic Manager(二) 基于权重与基于优先级的路由策略为我们的Web项目提供负载均衡

    一,引言 上一片文章我们使用 Azure Traffic Manager 分发用户请求,同时演示了两种路由策略,“Performance”,“Geographic”的两种方式,今天我们继续讲解 Tra ...

  9. MySQL中的循环

    MySQL中的三中循环 while . loop .repeat 求 1-n 的和第一种 while 循环 : /* while循环语法: while 条件 DO 循环体; end while; */ ...

  10. tomcat 认证爆破之custom iterator使用

    众所周知,BurpSuite是渗透测试最基本的工具,也可是神器,该神器有非常之多的模块:反正,每次翻看大佬们使用其的骚操作感到惊叹,这次我用其爆破模块的迭代器模式来练练手[不喜勿喷] 借助vulhub ...