探秘 RxJS Observable 为什么要长成这个样子?!

简介: 我们都知道 RxJS Observable 最基础的使用方法:是建立 Observable,即调用 .create API

image.png

它大概长什么样子



我们都知道 RxJS Observable 最基础的使用方法:是建立 Observable,即调用 .create API


使用方法如下🌰:


var observable = Rx.Observable
  .create(function(observer) {
    observer.next('Hi');
    observer.next('Jimmy');
  })
observable.subscribe(
    value => {console.log(value)}
)
// Hi
// Jimmy

image.png


创建一个 observable 实例后,当它调用 .subscribe,会将 .next(params) 的信息传到 value 并执行这个 function


next 的过程是同步的,但,也可以是异步的,比如🌰:


var observable = Rx.Observable
  .create(function(observer) {
    observer.next('Hi');
    observer.next('Jimmy');
    setTimeout(() => {
      observer.next('async');
    }, 30)
  })
console.log('start');
observable.subscribe(
  value => {console.log(value)}
);
console.log('end');
// start
// Hi
// Jimmy
// end
// async


其中,observer,除了 next 方法,


还有:

  • complete 方法
  • error 方法:


用法如下🌰:

var observable = Rx.Observable
  .create(function(observer) {
                observer.next('Hi');
                observer.next('Jimmy');
                observer.complete();
                observer.next('long time no see');
  })
observable.subscribe(
    value => { console.log(value); },
    error => { console.log('Error: ', error); },
    complete => { console.log('complete') },
)
// Hi
// Jimmy
// complete


complete 执行后,next 就会自动失效,不会再继续打印出 "long time no see";

而 error 用于捕获 try catch 中 throw 的 'msg';

...
                try {
                    observer.next('Hi');
                    observer.next('Jimmy');
                    throw 'some exception';
                } catch(e) {
                    observer.error(e)
                }
                ...


为什么这么写



那它为什么要长成这个样子呢?Why?

image.png


我就写这样,它不香?

function fn1(){
    console.log('Hi')
    console.log('Jimmy')
    return false
}
fn1()


为什么要改写为类似这样:


function f1(cb){
    cb('Hi')
    cb('Jimmy')
    return false
}
fn1(value => console.log(value))


本瓜最终理解为,其最具重要意义的一点是:为了将【数据】和【操作数据的行为】分开!!


实际上,分离【常量数据】和【操作方法】(也可以叫作计算方法/计算函数)是函数式编程的重要思想 —— 所有所需所求,都是通过计算得来的,而不改变变量的值;

(将一个函数作为参数传入另外一个函数,也就是 FP 中常提的高阶函数;)

过去,如果不将数据和操作分开,它是这样的:


function f1(){
    console.log('Hi')
    console.log('Jimmy')
    return false
}
function f2(){
    alert('Hi')
    alert('Jimmy')
    return false
}
fn1()
fn2()


在函数体内,是命令式的代码风格,将 A 数据这样操作,再将 B 数据这样操作;

而现在,将数据和操作分开,它是这样的:


function stream(cb){ 
    cb('Hi')
    cb('Jimmy')
    return false
}
stream(value => console.log(value))
stream(value => alert(value))


stream 函数体内,声明了 A 数据、B 数据...... 是一个数据流;

具体怎么再操作这些数据,就再看【操作方法】是什么;

这样做还有另外一个很大的 好处,就是:

对于多个数据流和多个操作,它的代码结构会更清晰,耦合度更低,拓展性更高;


它会是这样的:

class Stream {
  constructor(behavior) {
    this.behavior = behavior
  }
  doHandle(cb) {
    this.behavior(cb)
  }
}
const handleStream1 = new Stream(cb => {
   cb('Hi')
   cb('Jimmy')
})
const handleStream2 = new Stream(cb => {
   cb('Bye')
   cb('Lily')
})
/********  以上是数据流声明  ********/
/********     以下是操作     ********/
// 对数据流 1,做第一种操作
handleStream1.doHandle(
  value => console.log(value)
)
// 对数据流 1,做第二种操作
handleStream1.doHandle(
  value => alert(value)
)
// 对数据流 2,做第一种操作
handleStream2.doHandle(
  value => console.log(value)
)
// 对数据流 2,做第二种操作
handleStream2.doHandle(
  value => alert(value)
)

特别说明:以上的操作( consolealert) 只是一种最简单的替代操作的示意,实际代码中,会复杂的多(对数据进行复杂计算等等)。

我敲,这 Stream 不就是 Observable 吗!cb 不就是 observerdoHandle 不就是 subscribe!!


小结:

毫无疑问,Observable 还有更多神奇的妙用,本篇理解不过管中窥豹,但想要强调的重点即是:将数据声明和数据操作分离,是函数式编程中提高代码可读性的重要特性;至于数据形成的数据管道,以及常用管道操作,后面会继续介绍(关注不迷路)~~

不知道各位对 Observable 有怎样的理解,欢迎评论留言讨论。

我是掘金安东尼,输出暴露输入,技术洞见生活,下次见~~


相关文章
|
6月前
|
JavaScript 前端开发 调度
15_Rxjs
15_Rxjs
49 0
|
存储 缓存 JavaScript
RxJS系列03:主题 Subjects
RxJS系列03:主题 Subjects
111 1
|
JavaScript 前端开发 算法
RxJS系列06:测试 Observable
RxJS系列06:测试 Observable
106 0
|
前端开发 JavaScript API
RxJS系列01:响应式编程与异步
RxJS系列01:响应式编程与异步
198 0
|
存储 JavaScript 前端开发
RxJS系列02:可观察者 Observables
RxJS系列02:可观察者 Observables
|
前端开发 API 开发工具
MobX 源码解析-observable #86
MobX 源码解析-observable #86
114 0
|
前端开发 JavaScript API
继续解惑,异步处理 —— RxJS Observable
Observable 可观察对象是开辟一个连续的通信通道给观察者 Observer,彼此之前形成一种关系,而这种关系需要由 Subscription 来确立,而在整个通道中允许对数据进行转换我们称为操作符 Operator。
|
JavaScript 前端开发
三连弹!原生实现异步处理利器 —— Observable
本篇带来用原生实现 Observable,一探内部究竟!!
|
JavaScript 前端开发 调度
你会用RxJS吗?【初识 RxJS中的Observable和Observer】
概念 RxJS是一个库,可以使用可观察队列来编写异步和基于事件的程序的库。 RxJS 中管理和解决异步事件的几个关键点: Observable: 表示未来值或事件的可调用集合的概念。 Observer: 是一个回调集合,它知道如何监听 Observable 传递的值。 Subscription: 表示一个 Observable 的执行,主要用于取消执行。 Operators:** 是纯函数,可以使用函数式编程风格来处理具有map、filter、concat、reduce等操作的集合。
146 0
|
前端开发 JavaScript API
Rxjs源码解析(一)Observable
学习一个库最好的方法就是看其源码,理解其 api 的调用原理,用起来自然也就很清楚自己到底在干什么了,秉持着此观念,为了更好地理解 rxjs,抽空将其源码看了一遍 本文章不会刻意涉及概念性的东西,主线就是解读源码,并在恰当的时候给出一些小例子,源码基于 rxjs v7.4.0 版本
332 0