RxJS沉浸式入门教程

什么是RxJS【Reactive Extensions for JavaScript】

首先RxJS是一个库,是针对\color{#13c078}{异步数据流}编程工具,当然Angular引入RxJS就是让异步更加简单,更加可控,在开始RxJS之前,我们先来了解一下Reactive Programming,其本质就是使用\color{#13c078}{流(stream)}的一种编程方式。

什么是流呢?

首先看示例:

例子:有变量 a和b,  b 永远等于 a + 1
// 基础写法
let a = 1;
b = a + 1;
console.log(b)
a = 2;
b = a + 1; // 每次a变化,都要执行一次b = a+ 1
console.log(b)

 
// 使用流来解决:
// 用 stream 描述 a 与 b 的关系。
const a$ = new Subject(); 
const b$ = a$.pipe(map( x=> x + 1)); 
b$.subscribe(console.log); 
a$.next(1); 
a$.next(2);

虽然在这里我们还不认识Subject是个什么,但是通过这个例子我们可以了解到,所谓流/stream,就是数据基于事件(event)变化的整体。\color{#13c078}{stream = data + event},只要a变化,那么b就会自动跟着变化。

流的最大的好处是,能够监听数据的变化,执行相应的操作,从而最大可能性的减小性能的开销。对于rxjs而言任何东西都可以是streams

RxJS中的核心概念(Observable 、Observer 、Subscription、Subject)

在Angular项目中我们在调用接口的时候,常用的调用方式是:

this._exampleService.reportCurrentOpenTab$
       .subscribe((res: number) => {
         this.currentIndex = res;
       })
 //this._exampleService.reportCurrentOpenTab$ 这是返回observable,他可以是api的调用,可以是事件的调用等等

我们可以把上述的调用方式抽象一下为\color{#13c078}{observable.subscribe(observer)} 在这里我们认识到了两个新的事物分别是Observable和Observer,以及这个方法调用的返回对象,返回的是一个Subscription对象的实例化,接下来我们逐一介绍这些核心概念。

Observable

Observable是RxJS中最核心的一个概念,它的本质就是“Observable is a function to generate values”,首先它是一个\color{#13c078}{函数},也就是说它是\color{#13c078}{数据源头,是数据生产者},一般我们会在变量末尾加$表示Observable类型的对象。

// 此函数定义了setInterval 每两秒产生一个 value的功能
const observable$ = (observer) => {
    let counter = 0;
    const id = setInterval(() => observer.next(counter++), 2000);
}
// 因为Observable是个对象,所以需要调用才可以执行
observable$({ next: (val) => console.log(val) });

函数中会定义 value 的生成方式,函数调用时,observer.next 来执行在observer 中定义的行为,比如上述示例中的counter++。

  1. 必须调用(订阅)才会被执行
  2. observable 被调用后,必须能被关闭,否则会一只运行下去
  3. 对于同一个 observable,在不同的地方 subscribe,是无关的。function 执行多次,互相没有关联是一致的。

Observer

它是\color{#13c078}{观察者,数据使用者,数据消费者}。它是一个有三个回调函数的\color{#13c078}{对象},每个回调函数对应三种Observable发送的通知类型(next, error, complete),observer表示的是对序列结果的\color{#13c078}{处理方式}。在实际开发中,如果我们提供了\color{#13c078}{一个回调函数}作为参数,subscribe会将我们提供的函数参数作为\color{#13c078}{next}的回调处理函数。next决定传递一个什么样的数据给观察者

let observer = {
next: data => console.log('data'); // next表示数据正常流动,
error: err=> console.log('err'); // error表示流中出错
complete: () => console.log('complete') // complete表示流结束
}
// error和complete只会触发一个,但是可以有多个next

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

订阅 :Observer 通过 Observable 提供的 subscribe() 方法订阅 Observable。
发布 :Observable 通过回调 Next 方法向 Observer 发布事件。

Subscription

\color{#13c078}{observable.subscribe()}:是一个Subscription对象。在 observable.subscribe 内部,他会创建一个观察者对象并使用第一个回调函数作为next的处理方法。Subscription就是表示Observable 的执行,可以被清理。这个对象最常用的方法就是 \color{#13c078}{unsubscribe}方法。同时,它还有 \color{#13c078}{add}方法也可以使我们取消多个订阅,所以我们在Angular项目中会看到如下代码,去取消订阅:

    // 定义Subscription类型的数组同于存放页面中的observable订阅过程中产生的Subscription
    public subscriptions: Subscription[] = [];
    
    public ngOnInit(): void {
     this.subscriptions.push( // 将Subscription push到定义的数组中去
       this._layoutStatusService.reportCurOpenTab$
         .subscribe((res: number) => {
       this.currentIndex = res;
     })
 );
}

   // 在Angular的销毁生命周期中进行取消订阅,已达到优化性能的作用
   public ngOnDestroy(): void {
     this.subscriptions.forEach((subscription) => subscription.unsubscribe());
   }

我们可以用生活中 “订阅报刊” 的例子,再来理解一下订阅关系:
报刊生产商【observable】
市民【observer】

市民打电话给报刊生产商进行订阅【subscribe】,市民就会在报刊商送报纸(next)成功的时候收到报纸,这个时候市民是不用关心报纸是怎么生产和运送的,当然当报刊生产商可能会意外的导致送报纸失败(error),而等成功之后,报刊生产商还是会把这次送报标记为已送达(complete)

Subject

Subject是\color{#13c078}{特殊的observable}:我们可以像订阅任何observable一样去订阅subject。
Subject是\color{#13c078}{观察者}: 它有next(v),error(e),和complete()方法,如果我们需要给subject提供新值,只要调用\color{#13c078}{next(v)},它会将值多播给已注册监听该subject的观察者。
所以: Subject既是Observable,也是观察者(可以多个)

Subject与Observable的区别:
  • Subject是\color{#13c078}{多播的}【他可以将值多播给多个观察者】
  • 普通的Observble是\color{#13c078}{单播的}【每个已经订阅的观察者(observer)都拥有observable的独立执行,上述Observble的介绍也有提及】
Subject中常用的子类(ReplaySubject、AsyncSubject、BehaviorSubject)

这里针对Subject的这三个衍生子类,只做简单的介绍,在后续开发过程中如果需要使用该方法,请自行查阅官网,进行进一步的深入了解。

  • BehaviorSubject: 会保存\color{#13c078}{最先}发送数据,当被订阅的时,会立即使用这个最新【最先的】数据,然后会继续接收新的next的值。BehaviorSubject必须设置默认值,因为他有一个最新值(当前值)的概念。

  • ReplaySubject:会保存所有值,然后回放给最新的订阅者,当新的订阅发生的时候,会把上一次订阅的\color{#13c078}{所有值}都再次打印一遍~

  • AsyncSubject:只有当Observable执行完成时\color{#13c078}{complete()},它才会执行的\color{#13c078}{最后一个值}发送给观察者,就是说会保留流里最后一条数据,而且只会在数据流complete时候才会发送。

Subject的在Angular中的常见的作用:

可以在Angular通过service来实现不同组件,或者不同模块之间的传值

// 定义公共的用于数据存储的service,文件名是(eg:xampleStore.service.ts)
    @Injectable()
     export class ExampleStoreService {
      private currentTabNumber$ = new Subject<number>();
    }
        
// 此数据更改的逻辑,可以在任何需要更改的地方进行next相对应的值,文件名是 (eg:a.component.ts)
   this.ExampleStoreService.currentTabNumber$.next(1);
   
// 订阅接收到数据更改,并做下一步逻辑处理,文件名是(eg:b.component.ts)
  this.ExampleStoreService.currentTabNumber$
        .subscribe((res: number) => {
          this.currentIndex = res;
        })

RxJS的操作符(Operator)简介

operators是个纯函数,它的输入为observable,返回也observable。operators的本质是,描述从一个数据流到另一个数据流之间的关系,也就是observer到observable中间发生的转换,很类似于Lodash。
在RxJS中操作符有接近100个,不过在开发过程稿常用的也就十多个。

常见的运算符包含 map,filter,concat,flatmap,switchmap,forkjoin
在这里我们只调挑出forkjoin来讲解一下,其他的操作符可以自己去查阅使用
forkjoin主要是用于多个接口都返回的时候,才会返回结果,有点类似于promise中的promise.all,具体的用法如下:

forkJoin([
      this._aService.getData(),
      this._bService.getData()
    ]).subscribe(resArr => {
      // 此时的返回结果会被按顺序放在一个数组中
      aData = resArr[0];
      bData = resArr[1];
  }

Observable与Promise的比较:

observable.subscribe(function next(value) {});
promise.then(function resolve(value) {});

从方法的调用上感觉这两种处理异步的形式是相似的,实际上observable比promise强大很多。具体对比如下:

Observable:

  • RxJS 里面用的是 next() 和 subscribe()
  • next 会被调用多次,可以发射多个值
  • 但是observable在没有 subscribe 的情况下,是不是被执行的。
  • RxJS unsubscribe可以取消订阅
  • RxJS还提供了大量的操作符,可以方便出来各种流的情况

Promise:

  • Promise里面用的是 then() 和 resolve()
  • resolve 只会被调用一次,不能多次触发异步调用,
  • then并不会触发promise的执行。也就是说,一个 promise不管有没有then, 都会被执行。

Observable与Promise的转换

RxJS是有提供方法,用来进行observable与promise之间转换的,所以在其他使用promise的项目里完全可以使用RxJS去迭代开发。

const promise = observableFun$.toPromise() // 将observable转换成promise
const observable = observable.fromPromise(promiseFun) // 将promise类型函数转换成observable

通过一个搜索示例,再来回顾一下今天介绍的RxJS

普通方式:

<input id="inputControl"></input>
<script>
    var inputControl = document.querySelector('#inputControl'),
        timer = null,
        currentSearchText = '';

    text.addEventListener('keyup', (e) =>{
        clearTimeout(timer)
        timer = setTimeout(() => {
      
            currentSearchText = ''; 

            var searchText = e.target.value;
            $.ajax({
                url: `/search/${searchText}`,
                success: data => {
                    if (data.search === currentSearchText) {
                        // 渲染展示
                        render(data);
                    } else {
                        // ..
                    }
                }           
            });
        },300)
    })
</script>

使用流的方式:

var inputControl = document.querySelector('#inputControl');
var inputStream = fromEvent(inputControl, 'keyup')
                    .pipe(
                      .debounceTime(300) // 防抖动
                      .pluck('target', 'value') // 使用pluck操作符,获取输入的值
                      .switchMap(url => Http.get(url)) // 将当前输入流替换为http请求
                    .subscribe(data => render(data)); // 接收数据

总结

上述介绍的内容几乎都是Rxjs中很基础的东西,RxJS的入门确实有些难度,但是如果掌握了之后,你会发现RxJS这种响应式编程是真的香,本篇文章中并没有着重介绍RxJS中的操作符,如果想了解更多关于操作符可以参考RxJS的操作符介绍

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容