如何在不调用next的情况下设置新的BehaviorSubject值?

时间:2019-02-04 17:14:43

标签: angular rxjs behaviorsubject

我需要设置BehaviorSubject的值,而不会触发对任何订阅的下一个调用。

我尝试这样做:

this.mySubject = new BehaviorSubject(newVal);

但是也会删除所有订阅。

this.mySubject.value = newVal;

不起作用,因为.value是只读的。

有什么办法可以做到这一点?

编辑:针对那些询问为什么我需要这样做的人...

这是一个Ionic 4应用程序,因此在根列表页面中调用了另外2个子页面(详细信息视图和编辑页面)。进入视图页面的第一个导航需要选择的模型。列表页面将此模型的初始值设置为BehaviorSubject,然后导航至视图页面。但是,此初始值的设置会触发列表的刷新,而我不想这样做。我只想初始化该值,然后在编辑页面对其进行更改的情况下进行监听,然后刷新我的列表。

2 个答案:

答案 0 :(得分:1)

我很想知道您为什么真正想要这样做,但是您可以考虑的一个选择是在值中包含一个标志,该标志指示是否应传播更新,然后让所有订阅都从主题的过滤视图。

this.mySubject.next({propagate: false, value: 42});

makeObservable() {
   return this.mySubject.pipe(filter(x => x.propagate));
}

答案 1 :(得分:0)

BehaviorSubject并不那么复杂。您可以创建自己的实现:)

import { Subject } from './Subject';
import { Subscriber } from './Subscriber';
import { Subscription } from './Subscription';
import { SubscriptionLike } from './types';
import { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';

/**
 * A variant of Subject that requires an initial value and emits its current
 * value whenever it is subscribed to.
 *
 * @class BehaviorSubject<T>
 */
export class BehaviorSubject<T> extends Subject<T> {

  constructor(private _value: T) {
    super();
  }

  get value(): T {
    return this.getValue();
  }

  /** @deprecated This is an internal implementation detail, do not use. */
  _subscribe(subscriber: Subscriber<T>): Subscription {
    const subscription = super._subscribe(subscriber);
    if (subscription && !(<SubscriptionLike>subscription).closed) {
      subscriber.next(this._value);
    }
    return subscription;
  }

  getValue(): T {
    if (this.hasError) {
      throw this.thrownError;
    } else if (this.closed) {
      throw new ObjectUnsubscribedError();
    } else {
      return this._value;
    }
  }

  next(value: T): void {
    super.next(this._value = value);
  }
}

只需复制上面的内容,或创建一个暴露setValue的子类

export class SettableBehaviorSubject<T> extends BehaviorSubject<T> {
  setValue(value: T): void {
    this._value = value;
  }
}

BehaviorSubject的来源在这里:https://github.com/ReactiveX/rxjs/blob/master/src/internal/BehaviorSubject.ts