Rxjs debounce 運算子在 SAP Spartacus 函式節流中的一個實際使用例子

注销發表於2021-04-17

在 window-ref.ts 的實現裡,定義了一個每隔 300 毫秒,透過 fromEvent 發射一個 resize event 的Observable:

/**
   * Returns an observable for the window resize event and emits an event
   * every 300ms in case of resizing. An event is simulated initially.
   *
   * If there's no window object available (i.e. in SSR), a null value is emitted.
   */
  get resize$(): Observable<any> {
    if (!this.nativeWindow) {
      return of(null);
    } else {
      return fromEvent(this.nativeWindow, 'resize').pipe(
        debounceTime(300),
        startWith({ target: this.nativeWindow }),
        distinctUntilChanged()
      );
    }
  }

加上 distingctUntilChanged 運算子後,能過濾掉完全一致的 resize event. 下面的例子,展示瞭如何使用 distingctUntilChanged,將數字序列裡重複的數字過濾掉。distingctUntilChanged 預設會將當前元素和前一元素做比較。

import { of } from 'rxjs';
import { distinctUntilChanged } from 'rxjs/operators';

of(1, 1, 2, 2, 2, 1, 1, 2, 3, 3, 4).pipe(
    distinctUntilChanged(),
  )
  .subscribe(x => console.log(x)); // 1, 2, 1, 2, 3, 4

下列例子展示瞭如何將自定義的比較邏輯,透過箭頭函式作為引數,傳入 distinctUntilChanged 裡。

import { of } from 'rxjs';
import { distinctUntilChanged } from 'rxjs/operators';

interface Person {
   age: number,
   name: string
}

of<Person>(
    { age: 4, name: 'Foo'},
    { age: 7, name: 'Bar'},
    { age: 5, name: 'Foo'},
    { age: 6, name: 'Foo'},
  ).pipe(
    distinctUntilChanged((p: Person, q: Person) => p.name === q.name),
  )
  .subscribe(x => console.log(x));

// displays:
// { age: 4, name: 'Foo' }
// { age: 7, name: 'Bar' }
// { age: 5, name: 'Foo' }

相關文章