この記事ではRxJSの『combineLatestオペレータ』について、以下の内容を図とサンプルコードを用いて分かりやすく解説します。
combineLatest
オペレータとはcombineLatest
オペレータの「使い方」combineLatest
オペレータの「特徴」combineLatest
オペレータとwithLatestFrom
オペレータの違いと使い分け
RxJSのcombineLatestとは
RxJSのcombineLatestは、複数のObservableを入力として取り、それぞれのObservableの最新の値を組み合わせて新しいObservableを出力するオペレータです。
以下にcombineLatest
を用いたサンプルコードを示します。
import { combineLatest, timer } from 'rxjs';
// 1000ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerOne$ = timer(1000, 1000);
// 1500ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerTwo$ = timer(1500, 1000);
// combineLatestを使用して、2つのObservableの最新の値を組み合わせて新しいObservableを出力する
const combined$ = combineLatest([timerOne$, timerTwo$]);
combined$.subscribe((value) => console.log(value));
// ログ
// [ 0, 0 ]
// [ 1, 0 ]
// [ 1, 1 ]
// [ 2, 1 ]
// [ 2, 2 ]
// [ 3, 2 ]
// [ 3, 3 ]
// [ 4, 3 ]
上記のサンプルコードでは、combineLatest
を使用して、2つのObservable(timerOne$
とtimerTwo$
)の最新の値を組み合わせて新しいObservableを出力しています。
あわせて読みたい
『timerオペレータ』については下記の記事で詳しく説明しています。興味のある方は下記のリンクからぜひチェックをしてみてください。 続きを見る【RxJS】timerオペレータとは?「使い方」や「プログラム例」などを解説!
combineLatest
の返り値は配列のObservableなので、以下のプログラムに示すように、配列の分割代入を使用することもできます。
import { combineLatest, timer } from 'rxjs';
// 1000ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerOne$ = timer(1000, 1000);
// 1500ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerTwo$ = timer(1500, 1000);
// combineLatestを使用して、2つのObservableの最新の値を組み合わせて新しいObservableを出力する
const combined$ = combineLatest([timerOne$, timerTwo$]);
// 配列の分割代入を使用
combined$.subscribe(([timerValueOne, timerValueTwo]) => {
console.log(`Timer One:${timerValueOne} Timer Two:${timerValueTwo}`);
});
// ログ
// Timer One:0 Timer Two:0
// Timer One:1 Timer Two:0
// Timer One:1 Timer Two:1
// Timer One:2 Timer Two:1
// Timer One:2 Timer Two:2
// Timer One:3 Timer Two:2
// Timer One:3 Timer Two:3
// Timer One:4 Timer Two:3
また、以下に示すように、combineLatest
は関数を用いて、Observableの形を変更することもできます。
import { combineLatest, timer } from 'rxjs';
// 1000ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerOne$ = timer(1000, 1000);
// 1500ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerTwo$ = timer(1500, 1000);
// combineLatestを使用して、2つのObservableの最新の値を組み合わせて新しいObservableを出力する
// 関数を用いて、組み合わせたObservableの形を変更することもできる
const combined$ = combineLatest([timerOne$, timerTwo$], (timerValueOne, timerValueTwo) => {
let timeradd = timerValueOne + timerValueTwo;
return `Timer One(${timerValueOne}) + Timer Two(${timerValueTwo}) = ${timeradd}`;
});
combined$.subscribe((value) => console.log(value));
// ログ
// Timer One(0) + Timer Two(0) = 0
// Timer One(1) + Timer Two(0) = 1
// Timer One(1) + Timer Two(1) = 2
// Timer One(2) + Timer Two(1) = 3
// Timer One(2) + Timer Two(2) = 4
// Timer One(3) + Timer Two(2) = 5
// Timer One(3) + Timer Two(3) = 6
// Timer One(4) + Timer Two(3) = 7
combineLatestの特徴
combineLatest
の特徴を以下に示します。
- 全てのObservableが値を発行するまで出力されない
combineLatest
は、組み合わせる全てのObservableが最初の値を発行するまで何も出力しません。そのため、いずれかのObservableが遅れて値を発行する場合には注意が必要です。
- 各Observableの発行がトリガーとなる
- 全てのObservableが一度値を発行した後は、どのObservableが値を発行しても、その最新の値を組み合わせて出力されます。
上記の特徴が分かるサンプルコードをこれから説明します。
全てのObservableが値を発行するまで出力されない
combineLatest
は、組み合わせる全てのObservableが最初の値を発行するまで何も出力しません。サンプルコードを以下に示します。
以下のプログラムを見てみましょう。
import { combineLatest, timer } from 'rxjs';
// 1000ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerOne$ = timer(1000, 1000);
// 1500ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerTwo$ = timer(1500, 1000);
// 3750ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerThree$ = timer(3750, 1000);
// combineLatestを使用して、3つのObservableの最新の値を組み合わせて新しいObservableを出力する
const combined$ = combineLatest([timerOne$, timerTwo$, timerThree$]);
combined$.subscribe((value) => console.log(value));
// ログ
// [ 2, 2, 0 ]
// [ 3, 2, 0 ]
// [ 3, 3, 0 ]
// [ 3, 3, 1 ]
// [ 4, 3, 1 ]
// [ 4, 4, 1 ]
// [ 4, 4, 2 ]
// [ 5, 4, 2 ]
// [ 5, 5, 2 ]
// [ 5, 5, 3 ]
// [ 6, 5, 3 ]
// [ 6, 6, 3 ]
combineLatest
は、全てのObservableから少なくとも一度はデータが発行されるまで、何も出力しません、このプログラムでは、3750ms後に最初の値を発行するObservable(timerThree$
)があります。したがって、Observable(timerThree$
)が発行されるまで、combineLatest
は何も出力しません。
各Observableの発行がトリガーとなる
全てのObservableが一度値を発行した後は、どのObservableが値を発行しても、その最新の値を組み合わせて出力されます。サンプルコードを以下に示します。
import { combineLatest, timer } from 'rxjs';
// 1000ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerOne$ = timer(1000, 1000);
// 1500ms後に0を出力し、その後1000ms秒ごとに「1→2→3→4→・・・」というデータを流すObservable
const timerTwo$ = timer(1500, 1000);
// combineLatestを使用して、2つのObservableの最新の値を組み合わせて新しいObservableを出力する
const combined$ = combineLatest([timerOne$, timerTwo$]);
combined$.subscribe((value) => console.log(value));
// ログ
// [ 0, 0 ]
// [ 1, 0 ]
// [ 1, 1 ]
// [ 2, 1 ]
// [ 2, 2 ]
// [ 3, 2 ]
// [ 3, 3 ]
// [ 4, 3 ]
この例では、timerTwo$
が1000msごと、timerOne$
が1500msごとに値を発行します。それぞれのObservableが値を発行するたびに、その最新の値を組み合わせて出力されていることがわかります。
補足
オペレーターの動作を理解する上で、以下のサイトが非常に便利です。発行されるObservableを移動させたりして、各オペレーターの動作を確認できます。
combineLatestオペレータとwithLatestFromオペレータの違いと使い分け
combineLatest
と似たオペレータにwithLatestFrom
があります。withLatestFrom
は、メインのObservableの値が発行された際に、サブのObservableから最新の値を取得して組み合わせて、新しいObservableを出力するオペレータです。
あわせて読みたい
『withLatestFromオペレータ』については下記の記事で詳しく説明しています。興味のある方は下記のリンクからぜひチェックをしてみてください。 続きを見る【RxJS】withLatestFromオペレータとは?「使い方」や「特徴」などを解説!
以下にcombineLatest
とwithLatestFrom
の違いを示します。
combineLatest | withLatestFrom | |
合成可能なObservable数 | 2つ以上 | 2つ (メインとサブのObservable) |
初回値を流す条件 | 全てのObservableから少なくとも1回値が流れた時 | サブのObservableから1回以上値が流れている状態で、メインのObservableから値が流れた時 |
2回目以降の値を流す条件 | どれか1つのObservableが値を流したとき | メインのObservableが値を流したとき |
用途 | 複数のObservableの状態を常に最新で追跡したい場合 | メインのObservableに基づいてデータを加工する際に使用 |
combineLatest
とwithLatestFrom
の使い分けを以下に示します。
combineLatest
を使うとき- 2つ以上のObservableを合成する場合
- 2つのObservableを合成する場合において、どちらかのObservableから値が流れた時に値を流したい場合
- 例: 複数のフォーム入力状態の最新値をリアルタイムに結合して表示したい。
withLatestFrom
を使うとき- 2つのObservableを合成する場合において、メインのObservableから値が流れた時のみ値を流したい場合
- 例: ボタンをクリックした際に他の状態(サブObservable)を参照したい
本記事のまとめ
この記事ではRxJSの『combineLatestオペレータ』について、以下の内容を説明しました。
combineLatest
オペレータとはcombineLatest
オペレータの「使い方」combineLatest
オペレータの「特徴」combineLatest
オペレータとwithLatestFrom
オペレータの違いと使い分け
お読み頂きありがとうございました。