0

我对一个超出正常限制的可观察流有一些特殊需要/要求,我不完全确定如何做到这一点:

基本上我有一个可观察的流,最初来自一个正常的事件,如下所示:

var someEventObservable = Observable.FromEventPattern<SomeEventHandler, SomeEventArgs>(
    handler => this.ColumnWidthChanged += handler,
    handler => this.ColumnWidthChanged -= handler)
    .Select(_ => Unit.Default);

现在,由于这些事件可以快速连续发生,我只需要知道它是否在给定的时间范围内至少发生一次,我通常会使用 .Throttle(),即像这样:

var someThrottledEventObservable = someEventObservable
    .Throttle(TimeSpan.FromMilliseconds(300));

但是我的实际要求更进一步:如果在该限制 TimeSpan / dueTime 内引发了一个事件,并且如果在第一个事件之后引发了另一个事件但仍在该到期时间之内,我希望受限制的流再次从 0 等待时间重新开始再等待 300 毫秒……如果引发了另一个事件,请再次重新开始/延长该时间……依此类推。只有在原始或重新启动的 TimeSpan(s)/dueTime 内没有引发其他事件时,才someThrottledEventObservable应该产生一个新的 Unit 实例。

我希望这是有道理的 - 但基本上我想要/需要一个受限制的事件流,只要源流在给定时间内停止产生新事件并且如果在该等待时间内发生新事件,受限制的流应该重新启动等待。

或者:在持续的事件“风暴”中,仅 .Throttle() 就会每 300 毫秒产生一个新单元(在上面的示例中),但是每当一个或多个事件被触发但在 300 毫秒内没有发生新事件时,我都想要一个新单元之后的冷却期。

我该怎么做?

4

1 回答 1

0

正如@nikoniko 已经提到的那样,节流阀可以解决问题。

using System;
using System.Reactive.Linq;

namespace Printing {
class Program {
    static void Main(string[] args) {
        var source = Observable.Interval(TimeSpan.FromMilliseconds(333))
            .Do(i => Console.WriteLine($"new item: {i}"));
        var sampling = source.Throttle(TimeSpan.FromSeconds(1))
            .Do(i => Console.WriteLine($"sampled: {i}"));

        var subscription = sampling.Subscribe();

        Console.ReadLine();

        subscription.Dispose();

        Console.ReadLine();
    }
}

导致什么都没有,因为来自源的事件以两个高频率到达。但是如果源需要更多时间来传递一个元素,那么节流中给出的时间跨度:

using System;
using System.Reactive.Linq;

namespace Printing {
    class Program {
        static void Main(string[] args) {
            var source = Observable.Interval(TimeSpan.FromSeconds(1.2))
                .Do(i => Console.WriteLine($"{DateTime.Now.ToShortTimeString()}: new item: {i}"));
            var sampling = source.Throttle(TimeSpan.FromSeconds(1))
                .Do(i => Console.WriteLine($"{DateTime.Now.ToShortTimeString()}:  {i}"));

            var subscription = sampling.Subscribe();

            Console.ReadLine();

            subscription.Dispose();

            Console.ReadLine();
        }
    }
}

限制时间结束后会出现结果。如您所见,在触发源中的事件后的第二秒,它将出现在结果中。

08:32:26: new item: 0
08:32:27: throttle 0
08:32:28: new item: 1
08:32:29: throttle 1
08:32:30: new item: 2
08:32:31: throttle 2
08:32:32: new item: 3
08:32:33: throttle 3
08:32:34: new item: 4
08:32:35: throttle 4
08:32:36: new item: 5
08:32:37: throttle 5
于 2017-09-22T06:34:51.427 回答