Reactive Extensions同步订阅

时间:2014-07-16 05:09:17

标签: .net system.reactive reactive-programming

有人可以帮我对IObserver进行同步订阅,这样调用方法就会阻塞,直到订阅完成。 例如:

发布商

public static class Publisher {
public static IObservable<string> NonBlocking()
    {
        return Observable.Create<string>(
            observable =>
            {
                Task.Run(() =>
                {
                    observable.OnNext("a");
                    Thread.Sleep(1000);
                    observable.OnNext("b");
                    Thread.Sleep(1000);
                    observable.OnCompleted();
                    Thread.Sleep(1000);
                });

                return Disposable.Create(() => Console.WriteLine("Observer has unsubscribed"));
            });
    }

}

订户

public static class Subscriber{
public static bool Subscribe()
    {
        Publisher.NonBlocking().Subscribe((s) =>
        {
            Debug.WriteLine(s);
        }, () =>
        {
            Debug.WriteLine("Complete");
        });
        // This will currently return true before the subscription is complete
        // I want to block and not Return until the Subscriber is Complete
        return true;
    }

}

1 个答案:

答案 0 :(得分:6)

您需要使用System.Reactive.Threading.Task

将您的观察力转​​变为任务......

var source = Publisher.NonBlocking()
    .Do(
        (s) => Debug.WriteLines(x),
        ()  => Debug.WriteLine("Completed")
    )
    .LastOrDefault()
    .ToTask();

Do(...).Subscribe()就像Subscribe(...)一样。所以Do只会增加一些副作用。

LastOrDefault就在那里,因为Task创建的ToTask只会等待来自源Observable的第一个项目,如果没有项目产生,它将会失败(扔)。因此,LastOrDefault有效地导致Task等到源完成后,无论它产生什么。

所以在我们完成任务后,请等待它:

task.Wait(); // blocking

或使用async / await:

await task; // non-blocking

编辑:

Cory Nelson提出了一个很好的观点:

在最新版本的C#和Visual Studio中,您实际上可以await IObservable<T>。这是非常很酷的功能,但它的工作方式与等待Task的方式略有不同。

等待任务时,会导致任务运行。如果多次等待任务的单个实例,则该任务将仅执行一次。可观察量略有不同。您可以将observable视为具有多个返回值的异步函数...每次订阅observable时,observable / function都会执行。因此这两段代码有不同的含义:

等待观察者:

// Console.WriteLine will be invoked twice.
var source = Observable.Return(0).Do(Console.WriteLine);
await source; // Subscribe
await source; // Subscribe

通过任务等待观察:

// Console.WriteLine will be invoked once.
var source = Observable.Return(0).Do(Console.WriteLine);
var task = source.ToTask();
await task; // Subscribe
await task; // Just yield the task's result.

所以,从本质上讲,等待Observable就是这样的:

// Console.WriteLine will be invoked twice.
var source = Observable.Return(0).Do(Console.WriteLine);
await source.ToTask(); // Subscribe
await source.ToTask(); // Subscribe

但是,await observable语法在Xamerin Studio中不起作用(截至本文撰写时)。如果您使用的是Xamerin Studio,我强烈建议您在最后一刻使用ToTask来模拟Visual Studio await observable语法的行为。