Sobes.tech
Back to questions
Middle
4

Задача по RxJS: есть Observable, на который подписываются три раза в шаблоне. Как сделать одну подписку вместо трёх, не используя async pipe напрямую? Чем Subject и BehaviorSubject отличаются от обычного Observable (горячий vs холодный поток)?

Companies where asked
Цифротех

Answer from AI

sobes.tech AI

Если в шаблоне Angular трижды подписываются на один и тот же Observable, это может привести к тройному выполнению источника данных, что неэффективно. Чтобы сделать одну подписку вместо трёх, можно использовать оператор shareReplay или создать Subject/BehaviorSubject, который будет мультикастить значения.

Пример с shareReplay:

const source$ = this.http.get('/api/data').pipe(
  shareReplay(1) // кэширует последний эмит и делится им с подписчиками
);

Теперь несколько подписчиков получат одно и то же значение без повторных запросов.

Если не использовать async pipe, можно подписаться один раз в компоненте и передавать данные в шаблон через переменную.


Отличия Subject и BehaviorSubject от обычного Observable:

  • Observable — холодный поток: источник данных запускается заново для каждого подписчика.
  • Subject — горячий поток: мультикастит значения всем подписчикам, источник запускается один раз.
  • BehaviorSubject — это Subject с хранением текущего значения. При подписке сразу отдаёт последнее значение, даже если подписчик пришёл позже.

Таким образом, Subject и BehaviorSubject позволяют разделять один источник данных между несколькими подписчиками без повторного запуска источника.