3

CreateRegistryObservable我一直在研究的以下课程的方法中,我无法弄清楚如何IDisposable干净地处理下一行分配的内容。问题在于,我从被Action赋值的onComplete委托中引用它,该委托捕获具有 {null} 初始化的声明的数组变量,并且调用最后返回的内容的分配在调用Schedule()时没有反映本次行动。如何干净地处理调度程序?

scheduler[0] = Scheduler.CurrentThread.Schedule(iterator);

public class RegistryKeyMonitor<T> : IObservable<T>
{
    private readonly RegistryKey _registryKey;
    private readonly string _name;
    private readonly TimeSpan _timeSpan;
    private readonly ILog _logger = LogManager.GetLogger("RegistryKeyMonitor");

    public RegistryKeyMonitor(RegistryKey registryKey, string name, TimeSpan timeSpan)
    {
        _registryKey = registryKey;
        _name = name;
        _timeSpan = timeSpan;
    }

    public IDisposable Subscribe(IObserver<T> observer)
    {
        var sub =CreateRegistryObservable(_registryKey, _name, _timeSpan)
            //.Catch(Observable.Empty<object>())
            .Where(obj => obj != null)
            .Cast<T>();
        return sub.Subscribe(observer);

    }

    private IObservable<object> CreateValueChangeObservable(RegistryKey registryKey, string name, object initialValue, 
        TimeSpan timeSpan, CancellationToken token)
    {
        return Observable.Create<object>(o =>
            {
                var scheduler = Scheduler.Immediate;

                Action<object, Action<object, TimeSpan>> iterator = (current, self) =>
                    {
                        try
                        {
                            if (token.IsCancellationRequested) 
                            {   
                                o.OnCompleted();
                                return;
                            }
                            var value = registryKey.GetValue(name);
                            _logger.DebugFormat("{0} == registryKey.GetValue(name)", value);
                            if (current == null && value != null ||
                                current != null && !current.Equals(value))
                            {
                                _logger.DebugFormat("passing data {0}", value);
                                o.OnNext(value);
                            }
                            else
                                self(value, timeSpan);
                        }
                        catch (Exception e)
                        {
                            o.OnError(e);
                        }
                    };

                return scheduler.Schedule(initialValue, timeSpan, iterator);
            });
    }

    public IObservable<object> CreateRegistryObservable(RegistryKey registryKey, string name, TimeSpan timeSpan)
    {

        return Observable.Create<object>(o =>
            {
                var cancel = new CancellationDisposable();

                return new CompositeDisposable(cancel, 
                    NewThreadScheduler.Default.Schedule(() =>
                        {
                            IDisposable[] scheduler = {null};
                            var currentStateSubscription = new SerialDisposable();
                            object currentValue = null;

                            Action<Action> iterator = self =>
                                currentStateSubscription.Disposable =
                                CreateValueChangeObservable(registryKey, name, currentValue,
                                                            timeSpan, cancel.Token)
                                    .Subscribe(value =>
                                    {
                                        currentValue = value;
                                        self();
                                        o.OnNext(value);
                                    },
                                    o.OnError, 
                                    ()=> {
                                            currentStateSubscription.Dispose();
                                            scheduler[0].Dispose();
                                    }
                                    );
                            scheduler[0] = Scheduler.CurrentThread.Schedule(iterator);
                    }));
            });
    }
}
4

1 回答 1

0

我自己也想通了,如下。

public IObservable<object> CreateRegistryObservable(RegistryKey registryKey, string name, TimeSpan timeSpan)
    {

        return Observable.Create<object>(o =>
            {
                var cancel = new CancellationDisposable();

                var currentStateSubscription = new SerialDisposable();

                object currentValue = null;

                return new CompositeDisposable(cancel, NewThreadScheduler.Default.Schedule(
                    self => currentStateSubscription.Disposable =
                        CreateValueChangeObservable(registryKey, name, currentValue,
                                                    timeSpan, cancel.Token)
                            .Subscribe(value =>
                            {
                                currentValue = value;
                                self();
                                o.OnNext(value);
                            },
                            o.OnError, 
                            currentStateSubscription.Dispose
                            )
                    ));
            });
    }
于 2013-09-22T15:37:50.980 回答