6

[这个问题属于反应式扩展(Rx)领域]

需要在应用程序重新启动时继续的订阅

int nValuesBeforeOutput = 123;

myStream.Buffer(nValuesBeforeOutput).Subscribe(
    i => Debug.WriteLine("Something Critical on Every 123rd Value"));

现在我需要序列化和反序列化此订阅的状态,以便下次启动应用程序时,缓冲区计数不会从零开始,而是从应用程序退出之前的缓冲区计数开始

  • 在这种情况下,您如何保持 IObservable.Subscribe() 的状态并稍后加载它?
  • 是否有在 Rx 中保存观察者状态的通用解决方案?



从答案到解决方案

基于 Paul Betts 方法,这是一个半通用化的实现,在我的初始测试中有效

采用

int nValuesBeforeOutput = 123;

var myRecordableStream = myStream.Record(serializer);
myRecordableStream.Buffer(nValuesBeforeOutput).ClearRecords(serializer).Subscribe(
    i => Debug.WriteLine("Something Critical on Every 123rd Value"));

扩展方法

    private static bool _alreadyRecording;

    public static IObservable<T> Record<T>(this IObservable<T> input,
                                           IRepositor repositor) 
    {
        IObservable<T> output = input;
        List<T> records = null;
        if (repositor.Deserialize(ref records))
        {
            ISubject<T> history = new ReplaySubject<T>();
            records.ForEach(history.OnNext);
            output = input.Merge(history);
        }
        if (!_alreadyRecording)
        {
            _alreadyRecording = true;
            input.Subscribe(i => repositor.SerializeAppend(new List<T> {i}));
        }
        return output;
    }

    public static IObservable<T> ClearRecords<T>(this IObservable<T> input,
                                                 IRepositor repositor)
    {
        input.Subscribe(i => repositor.Clear());
        return input;
    }

笔记

  • 这不适用于存储依赖于生成的值之间的时间间隔的状态
  • 您需要一个支持序列化 T 的序列化器实现
  • _alreadyRecordingmyRecordableStream如果您多次订阅,则需要
  • _alreadyRecording是一个静态布尔值,非常丑陋,如果需要并行订阅,可以防止扩展方法在多个地方使用 - 需要重新实现以备将来使用
4

1 回答 1

1

对此没有通用的解决方案,制作一个将是 NonTrivial™。您可以做的最接近的事情是使 myStream 成为某种重放 Observable(即,不是序列化状态,而是序列化 myStream 的状态并重做工作以使您回到原来的位置)。

于 2012-04-10T20:42:09.610 回答