我尝试使用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
...
我怎样才能做到这一点?
答案 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
接受一个流流并在前一个子流完成时订阅每个连续的子流,将所有流连接成一个单独的平流。