javascript - RxJS 减少一个 ReplaySubject

标签 javascript rxjs reactivex

我正在使用 ReactiveX/RxJS 版本。

假设我有一个 Rx.ReplaySubject,它每 2 秒发出一个包含 id 和带有值的数组的对象。我想减少这个值数组并得到它们的总和。

问题是 ReplaySubject 是一个热的可观察对象并且它永远不会完成,至少我不希望它完成,因为我想要每 2 秒该对象值的总和。但是为了使用 reduce 运算符,应该完成 observable。那么,我应该如何进行呢?

E.G 不工作代码:

var subject = new Rx.ReplaySubject();

subject.
  map(x => x.transactions).
  // Reduce never concludes because ReplaySubject instance is not completed
  reduce((v1, v2) => v1+v2, 0).
  subscribe(function (value) {
    console.log(value)
  });

setInterval(injectData, 2000);

function injectData () {
  subject.next({id: Date.now(), transactions: [
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)}
  ]});
}

最佳答案

考虑使用 Observable.prototype.scan() (RxJS documentation)
scan()reduce() 不同,基本上聚合一个 observable 并发出每个连续值仅在完成时发出结果。 (参见 scanreduce 的 Rx 解释)

使用 OP 代码的示例(这里是 fiddle ):

var subject = new Rx.ReplaySubject();

subject
  // note: use "selectMany" to flatten observable of observables
  .selectMany(x => Rx.Observable.fromArray(x.transactions))
  // note: use "scan" to aggregate values
  .scan((agg, val) => agg+val.value, 0)
  .subscribe(function (value) {
    console.log(value)
  });

setInterval(injectData, 2000);

function injectData () {
  subject.onNext({id: Date.now(), transactions: [
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)},
    {value: Math.round(Math.random() * 5000)}
  ]});
}

另一个例子:

由于 selectMany(),上面的代码为每个事务发出聚合。 .如果您只希望它每 2 秒发射一次,这是使用 reduce() 的好时机像这样(这是另一个 fiddle ):
subject
  // note: use "selectMany" to flatten observable of observables
  // note: using "reduce" inside here so that we only emit the aggregate
  .selectMany(x => 
    Rx.Observable
      .fromArray(x.transactions)
      .reduce((agg, val) => agg + val.value, 0)
  )
  // note: use "scan" to aggregate values
  .scan((agg, val) => agg+val, 0)
  .subscribe(function (value) {
    console.log(value)
  });

附加说明:

Rx Subjects 可以完成;您只需调用onCompleted()当你准备好时。如果你完成了你的主题,你仍然可以使用 reduce() .比较这个 fiddle与上面的那个。

关于javascript - RxJS 减少一个 ReplaySubject,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36075342/

相关文章:

javascript - Cordova 相机插件适用于模拟器,不适用于 Android 设备

node.js - Angular : Receive responses in order with the calls

c# - 为什么这些 Rx 序列乱序?

java - 等待多个可观察对象完成并返回不同数量的元素

javascript - ReactJS 中的浏览器检测

javascript - 两层应用程序中的 Sails.js socket.io。如何定义连接选项?

javascript - 为什么 RxJS Observable 不能在 Vue.js 应用程序 div 中工作?

mongodb - 使用JavaRx/rx-mongodb在Grails中下载文件

javascript - 更改文本框 onchange 下拉列表的输入类

使用 RxJS : Cannot read property 'call' of undefined 的 Angular 6 错误