我一直想了解这3个:

主题 BehaviorSubject ReplaySubject

我想使用它们,知道什么时候和为什么,使用它们的好处是什么,虽然我已经阅读了文档,观看教程和搜索谷歌,但我没有搞清楚这一点。

那么它们的目的是什么呢?如果是真实的情况,甚至不需要编写代码,那就最好不过了。

我更喜欢一个干净的解释,而不仅仅是“a+b => c您订阅了....”

谢谢你!


当前回答

这实际上归结为行为和语义。与一个

Subject - a subscriber will only get published values that were emitted after the subscription. Ask yourself, is that what you want? Does the subscriber need to know anything about previous values? If not, then you can use this, otherwise choose one of the others. For example, with component-to-component communication. Say you have a component that publishes events for other components on a button click. You can use a service with a subject to communicate. BehaviorSubject - the last value is cached. A subscriber will get the latest value upon initial subscription. The semantics for this subject is to represent a value that changes over time. For example a logged in user. The initial user might be an anonymous user. But once a user logs in, then the new value is the authenticated user state. The BehaviorSubject is initialized with an initial value. This is sometimes important to coding preference. Say for instance you initialize it with a null. Then in your subscription, you need to do a null check. Maybe OK, or maybe annoying. ReplaySubject - it can cache up to a specified number of emissions. Any subscribers will get all the cached values upon subscription. When would you need this behavior? Honestly, I have not had any need for such behavior, except for the following case: If you initialize a ReplaySubject with a buffer size of 1, then it actually behaves just like a BehaviorSubject. The last value is always cached, so it acts like a value changing over time. With this, there is no need for a null check like in the case of the BehaviorSubject initialized with a null. In this instance, no value is ever emitted to the subscriber until the first publishing.

因此,这实际上取决于您所期望的行为(至于使用哪一种)。大多数情况下,您可能想要使用BehaviorSubject,因为您真正想要表示的是“随时间变化的值”语义。但我个人不认为用1初始化ReplaySubject的替换有任何错误。

当您真正需要缓存行为时,您要避免使用普通Subject。例如,您正在编写路由保护或解析。您从该保护中获取一些数据,并将其设置在服务Subject中。然后在路由组件中订阅服务主题,以尝试获取在保护中发出的值。哦。价值在哪里?它已经被释放了,DUH。使用“缓存”主题!

参见:

什么是RxJS Subject和使用它们的好处?

其他回答

这实际上归结为行为和语义。与一个

Subject - a subscriber will only get published values that were emitted after the subscription. Ask yourself, is that what you want? Does the subscriber need to know anything about previous values? If not, then you can use this, otherwise choose one of the others. For example, with component-to-component communication. Say you have a component that publishes events for other components on a button click. You can use a service with a subject to communicate. BehaviorSubject - the last value is cached. A subscriber will get the latest value upon initial subscription. The semantics for this subject is to represent a value that changes over time. For example a logged in user. The initial user might be an anonymous user. But once a user logs in, then the new value is the authenticated user state. The BehaviorSubject is initialized with an initial value. This is sometimes important to coding preference. Say for instance you initialize it with a null. Then in your subscription, you need to do a null check. Maybe OK, or maybe annoying. ReplaySubject - it can cache up to a specified number of emissions. Any subscribers will get all the cached values upon subscription. When would you need this behavior? Honestly, I have not had any need for such behavior, except for the following case: If you initialize a ReplaySubject with a buffer size of 1, then it actually behaves just like a BehaviorSubject. The last value is always cached, so it acts like a value changing over time. With this, there is no need for a null check like in the case of the BehaviorSubject initialized with a null. In this instance, no value is ever emitted to the subscriber until the first publishing.

因此,这实际上取决于您所期望的行为(至于使用哪一种)。大多数情况下,您可能想要使用BehaviorSubject,因为您真正想要表示的是“随时间变化的值”语义。但我个人不认为用1初始化ReplaySubject的替换有任何错误。

当您真正需要缓存行为时,您要避免使用普通Subject。例如,您正在编写路由保护或解析。您从该保护中获取一些数据,并将其设置在服务Subject中。然后在路由组件中订阅服务主题,以尝试获取在保护中发出的值。哦。价值在哪里?它已经被释放了,DUH。使用“缓存”主题!

参见:

什么是RxJS Subject和使用它们的好处?

被点赞最多的回答显然是错误的:

如果你初始化一个ReplaySubject缓冲区大小为1,那么它的行为实际上就像一个BehaviorSubject


这并不完全正确;看看这篇关于两者区别的博客文章。例如,如果你订阅了一个完整的BehaviorSubject,你将不会收到最后一个值,但对于ReplaySubject(1),你将收到最后一个值。

这是一个不容忽视的重要区别:

const behavior = new behaviour subject (null); const replay = new ReplaySubject(1); behavior.skip(1)。订阅(v => console.log('BehaviorSubject:', v)); 重播。订阅(v => console.log('ReplaySubject:', v)); behavior.next (1); behavior.next (2); behavior.complete (); 的行为。订阅(v => console.log('晚期B订阅者:',v)); replay.next (1); replay.next (2); replay.complete (); 重播。订阅(v => console.log('后期R订阅者:',v));

查看这里的代码示例,它来自另一篇关于该主题的很棒的博客文章。

另一个不同之处在于,您可以使用BehaviorSubject的值getter来获取当前值。当您在某些情况下只需要当前值时,这是非常有用的。例如,当用户单击某个内容,而您只需要该值一次时。在这种情况下,你不需要订阅然后突然取消订阅。唯一需要的是:

BehaviorSubject bSubject = new BehaviorSubject<IBasket>(basket);

getCurrentBasketValue() {
  return this.bSubject.value;
}
     // ***********Subject  concept ***********
    let subject = new Subject<string>();


    subject.next("Eureka");
    subject.subscribe((data) => {
      console.log("Subscriber 1 got data >>>>> "+ data);
    });
    subject.subscribe((data) => {
      console.log("Subscriber 2 got data >>>>> "+ data);
    });

       // ********behaviour subject*********
    // Behavior subjects need a first value
let subject1 = new BehaviorSubject<string>("First value");


subject1.asObservable().subscribe((data) => {
  console.log("First subscriber got data behaviour subject>>>>> "+ data);
});
subject1.next("Second value")

Subject——订阅者只会在订阅之后获得已发布的值。 BehaviorSubject——新订阅者在订阅后立即获得最后发布的值或初始值。

正如在一些帖子中提到的,公认的答案是错误的,因为BehaviorSubject != ReplaySubject(1),这不仅仅是编码风格的偏好。

评论中经常提到“守卫”,这也是我发现重放主题用例最多的地方。更具体地说,如果你有一个类似于取(1)的情况你不只是想取初值。

请看下面的例子:

  ngOnInit() {
    const behaviorSubject = new BehaviorSubject<boolean>(null);
    const replaySubject = new ReplaySubject<boolean>(1);
    this.checkLoggedIn(behaviorSubject, 'behaviorSubject');
    this.checkLoggedIn(replaySubject, 'replaySubject');
    behaviorSubject.next(true);
    replaySubject.next(true);
  }

  checkLoggedIn($userLoggedIn: Observable<boolean>, id: string) {
    $userLoggedIn.pipe(take(1)).subscribe(isLoggedIn => {
      if (isLoggedIn) {
        this.result[id] = 'routed to dashboard';
      } else {
        this.result[id] = 'routed to landing page';
      }
    });
  }

结果是:

{
  "behaviorSubject": "routed to landing page",
  "replaySubject": "routed to dashboard"
}

在这种情况下,你显然需要一个ReplaySubject!工作代码:https://stackblitz.com/edit/replaysubject-vs-behaviorsubject?file=src%2Fapp%2Fapp.component.ts