Rx.Net:在SelectMany中调用多个IObservable

时间:2018-05-06 06:43:48

标签: system.reactive rx.net

请注意:这是之前发布的question的延续,但感兴趣的解决方案情况不同。

我试图对每个返回IObservable的方法进行多次调用,但是在SelectMany语句中返回的值是一个Task,因此以下的Subscribe语句不能编译

这是代码段

 var myWorkList = new List<MyWork>
                {
                    new MyWork(),// MyWork.Execute(data) returns IObservable
                    new MyWork()
                }.ToObservable();

 var results =
   myService
    .GetData(accountId)
    .SelectMany(data => myWorkList.ForEachAsync(r => r.Execute(data))
    .Subscribe(result =>
    {
        Console.WriteLine($"Result Id: {result.Id}");
        Console.WriteLine($"Result Status: {result.Pass}");
    });

2 个答案:

答案 0 :(得分:1)

您只想使用.SelectMany。试试这个:

var myWorkList = new List<MyWork>()
{
    new MyWork(),
    new MyWork()
}.ToObservable();

var query =
    from data in myService.GetData(accountId)
    from myWork in myWorkList
    from result in myWork.Execute(data)
    select result;

var results =
    query
        .Subscribe(result =>
        {
            Console.WriteLine($"Result Id: {result.Id}");
            Console.WriteLine($"Result Status: {result.Pass}");
        });

这是我的测试代码:

public static class myService
{
    public static IObservable<MyData> GetData(int x)
        => Observable.Return(new MyData());
}

public class MyWork
{
    public virtual IObservable<MyResult> Execute(MyData data)
    {
        return
            from isMatch in IsMatch(data)
            where isMatch
            select new MyResult() { Id = 1, Pass = true };
    }

    public IObservable<bool> IsMatch(MyData data)
    {
        return Observable.Return(true);
    }
}

public class MyResult
{
    public int Id;
    public bool Pass;
}

public class MyData { }

当我执行时,我得到了这个:

Result Id: 1
Result Status: True
Result Id: 1
Result Status: True

在您对上一个问题的评论中,我建议将其作为代表列表。方法如下:

var myWorkList = new Func<MyData, IObservable<MyResult>>[]
{
    md => new MyWork().Execute(md),
    md => new MyWork().Execute(md),
}.ToObservable();

var query =
    from data in myService.GetData(accountId)
    from myWork in myWorkList
    from result in myWork(data)
    select result;

你得到的结果相同。

答案 1 :(得分:1)

列表可以在MyWork - s的列表中声明 - 在此处不使用ToObservable

var myWorkList = new List<MyWork>
            {
                new MyWork(),// MyWork.Execute(data) returns IObservable
                new MyWork()
            };

然后,我们将myService.GetData返回的对象映射到myWorkList的元素,并将它们作为IObservable - s。

var observables = myService
         .GetData(accountId)
         .SelectMany(data => myWorkList.Select(r => r.Execute(data)));

现在你可以观察它们了。

一起 - 合并:

var subscription =
          observables
           .Merge()
           .Subscribe(result =>
           {
               ...
           });

或单独:

var subscriptions=
          observables
           .Select(obs => 
               obs.Subscribe(result =>
               {
                   ...
               }))
           .ToArray();

更新:后一种情况必须立即实现,以防止副作用(。ToArray())。