一次执行一个异步可观察

时间:2013-12-05 15:54:34

标签: c# .net task-parallel-library system.reactive

我尝试使用Observable.FromAsync将一些TPL异步整合到更大的Rx链中,就像在这个小例子中一样:

using System;
using System.Reactive.Linq;
using System.Threading.Tasks;

namespace rxtest
{
    class Program
    {
        static void Main(string[] args)
        {
            MainAsync().Wait();
        }

        static async Task MainAsync()
        {
            await Observable.Generate(new Random(), x => true,
                                      x => x, x => x.Next(250, 500))
                .SelectMany((x, idx) => Observable.FromAsync(async ct =>
                {
                    Console.WriteLine("start:  " + idx.ToString());
                    await Task.Delay(x, ct);
                    Console.WriteLine("finish: " + idx.ToString());
                    return idx;
                }))
                .Take(10)
                .LastOrDefaultAsync();
        }
    }
}

但是,我注意到这似乎同时启动所有异步任务,而不是一次启动它们,这会导致应用程序的内存使用量增加。 SelectMany似乎与Merge没有区别。

在这里,我看到这样的输出:

start:  0
start:  1
start:  2
...

我想看看:

start:  0
finish: 0
start:  1
finish: 1
start:  2
finish: 2
...

我怎样才能做到这一点?

1 个答案:

答案 0 :(得分:8)

使用SelectMany

Select更改为Concat
    static async Task MainAsync()
    {
        await Observable.Generate(new Random(), x => true,
                                  x => x, x => x.Next(250, 500))
            .Take(10)
            .Select((x, idx) => Observable.FromAsync(async ct =>
            {
                Console.WriteLine("start:  " + idx.ToString());
                await Task.Delay(x, ct);
                Console.WriteLine("finish: " + idx.ToString());
                return idx;
            }))
            .Concat()
            .LastOrDefaultAsync();
    }

编辑 - 我将Take(10)移动到链上,因为Generate不会阻止 - 所以它会阻止它逃跑。

Select将每个事件投影到一个流中,该流表示将在Subscription上启动的异步任务。 Concat接受一个流流并在前一个子流完成时订阅每个连续的子流,将所有流连接成一个单独的平流。