Вы когда-нибудь встречали такие операторы, как materialize и dematerialize в RxJS? А что насчет класса Notification? Вероятно, многие слышали, но не до конца представляли, где их можно применить на практике.
В этой статье я расскажу, что делают эти операторы и приведу несколько примеров, которые в будущем вам могут пригодиться.
Однажды, интереса ради, я просматривала документацию RxJS и заметила ранее для себя неизвестные операторы: materialize и dematerialize. Первый вопрос, который у меня возник: а где их использовать в реальном проекте?
Разберемся вместе?
Materialize
Для начала вспомним, какие типы значений может эмитить объект типа Observable: это next, error и complete. Если вы не помните, что это значит, здесь можно почитать.
Соответственно и про observer, набор коллбэков (onNext, onError, onComplete), тоже советую вспомнить.
Вот что говорится в документации о materialize операторе: «A function that returns an Observable that emits Notification objects that wrap the original emissions from the source Observable with metadata».
Сигнатура оператора:
materialize<T>(): OperatorFunction<T, Notification<T> & ObservableNotification<T>>
Получается, materialize оборачивает любое испускаемое observable значение в некий Notification объект и эмитит его как «next».
Чем здесь является Notification? Это обертка над observable значением, которая просто добавляет к нему некоторые метаданные:
class Notification<T> {
static createNext<T>(value: T)
static createError(err?: any)
static createComplete(): Notification<never> & CompleteNotification
constructor(kind: "N" | "E" | "C", value?: T, error?: any)
get hasValue: boolean
get kind: 'N' | 'E' | 'C'
get value?: T
get error?: any
observe(observer: PartialObserver<T>): void
do(nextHandler: (value: T) => void, errorHandler?: (err: any) => void, completeHandler?: () => void): void
accept(nextOrObserver: NextObserver<T> | ErrorObserver<T> | CompletionObserver<T> | ((value: T) => void), error?: (err: any) => void, complete?: () => void)
toObservable(): Observable<T>
}
Больше всего нас здесь интересует свойство kind. Именно в нем хранится первоначальный тип значения observable: N — next, E — error, C — complete.
Давайте посмотрим на визуализацию работы оператора:
Теперь на примере:
of(“one”) // поток эмитит одно значение и после завершается
.pipe(materialize()) // оборачиваем в Notification object
.subscribe(x => console.log(x)); // в подписку поочередно прилетает 2 Notification
Output:
{kind: “N”, value: “one”, error: undefined, hasValue: true} // Notification object
{kind: “C”, value: undefined, error: undefined, hasValue: false} // Notification object
Давайте разберемся:
поток эмитит "one" -> materialize конвертирует в Notification, где kind = "N" (next), а в value записывается передаваемое значение "one".
поток завершается -> materialize конвертирует в Notification, где kind = "C" (complete), а в value ничего не записывается, т.к. поток завершился.
Из-за того, что materiailze конвертирует значения в Notification объект, observable эмитит их как «next», и мы видим обернутый complete выброс в next обработчике observable.
Еще пример:
throwError(() => new Error(“error”)) // эмитим ошибку
.pipe(materialize()) // конвертируем в Notification
.subscribe(x => console.log(x));
Output:
{kind: “E”, value: undefined, error: Error, hasValue: false}
Опять же из-за того, что materiailze все оборачивает в Notification, даже ошибка попадает в next обработчик.
Dematerialize
Но что делать, если мы хотим вернуться к исходному выбросу observable? Здесь нам поможет противоположность materialize оператора — dematerialize, с его помощью происходит обратная конвертация.
Из документации: «Converts an Observable of Notification objects into the emissions that they represent».
Сигнатура оператора:
dematerialize<N extends ObservableNotification<any>>(): OperatorFunction<N, ValueFromNotification<N>>
Немного визуализации:
Работа оператора в коде:
// 1 пример
of(“one”) // эмитится одно значение, и завершается поток
.pipe(
materialize(), // конвертируем в Notification oбъект
dematerialize()) // конвертируем Notification в исходное значение потока
.subscribe(x => console.log(x));
Output: "one" // получили то, что изначально эмитил поток
// 2 пример
throwError(() => new Error(“error”)) // эмитим ошибку
.pipe(
materialize(), // конвертируем в Notification oбъект
dematerialize()) // конвертируем Notification в исходное значение потока
.subscribe(x => console.log(x));
Output: // ничего не логируется next обработчиком,
ERROR Error: error // видим обычную ошибку
А нужно ли оно мне?
Мы вкратце рассмотрели, что делают операторы materialize и dematerialize. Но где же они могут пригодиться?
Первое, где их можно использовать — во время дебага. Например, вам нужно добавить задержку во время эмита ошибки.
Если это сделать следующим образом:
throwError(() => new Error(“message”))
.pipe(delay(3000)) // добавляем задержку
.subscribe({
next: (x) => { console.log(`${x}`); },
error: (err) => { console.log(`${err}`); }
complete: () => { console.log('complete'); }
})
То в этом случае ошибку вы увидите моментально, задержка не сработает. Это происходит потому, что оператор delay работает только для next выбросов.
Но если использовать изученные нами операторы, то ошибка выводится после заданного времени задержки.
throwError(() => new Error('message'))
.pipe(
materialize(), // ошибка конвертируется в Notification объект, это позволяет нам применить delay y
delay(5000), // добавляем задержку
dematerialize()) // конвертируем обратно в ошибку
.subscribe({
next: (x) => { console.log(`${x}`); },
error: (err) => { console.log(`${err}`); },
complete: () => { console.log('complete'); },
});
Output: Error: message // сработал error обработчик через 5 секунд
Все из-за того, что ошибка конвертируется в Notification объект с помощью materialize, это позволяет нам применить оператор delay.
После (с помощью dematerialize оператора) мы получаем первоначальную ошибку.
Где же еще они пригодятся?
Скажем, нужно объединить 2 observable. Как только один из них завершится, то результирующий observable тоже должен завершиться.
Если реализовать это с помощью оператора merge:
const result$ = merge(this.first$, this.second$)
.subscribe({
next: (x) => { console.log(`${x}`); },
complete: () => { console.log('complete'); },
});
this.first$.next(1);
this.second$.complete(); // завершаем один из потоков
this.first$.next(2); // результирующий поток все еще не завершен
Output:
1
2
Мы не увидим в output "complete". Это случилось из-за того, что merge просто так работает: «The output Observable only completes once all input Observables have completed». И один из способов справиться с этим — использовать операторы materialize и dematerialize:
const result$ = merge(
this.first$.pipe(materialize()), // конвертируем в Notification 1 поток
this.second$.pipe(materialize()) // конвертируем в Notification 2 поток
)
.pipe(dematerialize()) // возвращаемся к исходному значению
.subscribe({
next: (x) => { console.log(`${x}`); },
complete: () => { console.log('complete'); },
});
this.first$.next(1); // Notification {kind: "N"}
this.second$.complete(); // завершаем один из потоков - Notification {kind: "C"}
this.first$.next(2); // этот next уже на сработает
Output:
1 // next от first$
complete // complete от second$ — завершился результирующий поток
Это то, чего мы добивались. Догадываетесь, почему сработало?
Мы знаем, что при использовании merge оператора результирующий поток не завершится при завершении только одного из переданных в него observable. А materialize оператор конвертирует любое значение observable в Notification объект, что воспринимается как «next» выброс.
Это позволило нам получить "complete" событие от this.second$ observable в pipe(...).
И уже с помощью dematerialize оператора мы получаем исходное значение observable — завершение потока.
Заключение
Я постаралась привести несколько примеров использования materialize/dematerialize, чтобы понять, на что они способны и где полезны.
Уверена, что эти операторы помогут вам лучше понимать и использовать все возможности RxJS, и вы найдете им интересные способы применения.
А вы раньше использовали их?
Полезные ссылки
https://rxjs.dev/api/operators/materialize
https://rxjs.dev/api/operators/dematerialize
https://rxjs.dev/api/index/class/Notification
https://docs.w3cub.com/rxjs/api/index/class/notification
https://docs.w3cub.com/rxjs/api/operators/materialize
https://docs.w3cub.com/rxjs/api/operators/dematerialize
https://docs.w3cub.com/rxjs/api/index/function/merge
Комментарии (7)
enkryptor
20.04.2022 14:42+2Оператор materialize() можно использовать для реализации показа лоадера/ошибки. Например, есть Observable, значение которого нужно отобразить в интерфейсе. Обычно в разметке это выглядит как {{ source$ | async }}
Но значение подгружается динамически с сервера, загрузка требует времени и может завершиться неуспешно. Хотелось бы показывать какой-то индикатор загрузки (спиннер или надпись "loading...").
Для решения этой задачи можно сделать {{ source$ | async as value else loadingTemplate }}, но проблема такого решения в том, что при получении ошибки пользователю будет показан бесконечный лоадер.
Хотелось бы останавливать лоадер и показывать ошибку если загрузка не удалась. Для обработки этих случаев идеально подходит объект Notification, который возвращает materialize()
Со стороны потребителя это может выглядеть и как кастомный аналог пайпа async, и как отдельный источник sourceStatus$, и как прямой байндинг на поля Notification.
Пример:
<ng-container source$ | async as notification else loading>
<span *ngIf="notification.hasValue">notification.value</span>
<span *ngIf="notification.kind === 'E'" class="error">Error: {{ notification.error }}</span>
</ng-container>
<ng-template #loading>
<span>Loading...</span>
</ng-template>
amakhrov
Первый случай с задержкой можно проще:
Второй - еще проще
Я использовал materialize чуть ли не один раз в жизни. И вот тоже сейчас смотрю - пожалуй, мой случай можно было реализовать через
defaultIfEmpty
, и было бы чуть проще.MiliyKot Автор
Привет!
1) Да, это один из вариантов, почему бы и нет
Вероятно, кому-то будет удобен мой способ)
2) Все же race и merge работают совершенно по-разному, и с race не удастся воспроизвести то поведение, которое у меня в примере, например:
Собственно, как и должен работать race - выигрывает в нашем случае subject1$, и следим далее мы только за ним (что заэмитилось первым, то и юзается)
очевидно, при complete subject2$ ничего не произойдет,
а это совсем не то, что я предлагала решить в статье)
amakhrov
Да, вы правы, а я был невнимателен. С race будет то же самое только в случае, если оба источника одноразовые (к примеру, запросы к серверу).
Пожалуй да, вариант с materialize самый простой. Даже удивительно, что в rxjs из коробки нет какого-то оператора для такого поведения.
MiliyKot Автор
Да, обычно можно найти оператор на любой случай, а тут изворачиваться приходится ????