Nesting await in Parallel.ForEach

后端 未结 9 1341
别跟我提以往
别跟我提以往 2020-11-22 01:01

In a metro app, I need to execute a number of WCF calls. There are a significant number of calls to be made, so I need to do them in a parallel loop. The problem is that th

9条回答
  •  遥遥无期
    2020-11-22 01:41

    Using DataFlow as svick suggested may be overkill, and Stephen's answer does not provide the means to control the concurrency of the operation. However, that can be achieved rather simply:

    public static async Task RunWithMaxDegreeOfConcurrency(
         int maxDegreeOfConcurrency, IEnumerable collection, Func taskFactory)
    {
        var activeTasks = new List(maxDegreeOfConcurrency);
        foreach (var task in collection.Select(taskFactory))
        {
            activeTasks.Add(task);
            if (activeTasks.Count == maxDegreeOfConcurrency)
            {
                await Task.WhenAny(activeTasks.ToArray());
                //observe exceptions here
                activeTasks.RemoveAll(t => t.IsCompleted); 
            }
        }
        await Task.WhenAll(activeTasks.ToArray()).ContinueWith(t => 
        {
            //observe exceptions in a manner consistent with the above   
        });
    }
    

    The ToArray() calls can be optimized by using an array instead of a list and replacing completed tasks, but I doubt it would make much of a difference in most scenarios. Sample usage per the OP's question:

    RunWithMaxDegreeOfConcurrency(10, ids, async i =>
    {
        ICustomerRepo repo = new CustomerRepo();
        var cust = await repo.GetCustomer(i);
        customers.Add(cust);
    });
    

    EDIT Fellow SO user and TPL wiz Eli Arbel pointed me to a related article from Stephen Toub. As usual, his implementation is both elegant and efficient:

    public static Task ForEachAsync(
          this IEnumerable source, int dop, Func body) 
    { 
        return Task.WhenAll( 
            from partition in Partitioner.Create(source).GetPartitions(dop) 
            select Task.Run(async delegate { 
                using (partition) 
                    while (partition.MoveNext()) 
                        await body(partition.Current).ContinueWith(t => 
                              {
                                  //observe exceptions
                              });
    
            })); 
    }
    

提交回复
热议问题