【问题标题】:How to process tasks as they complete -but each task requires different method to process task's result如何在任务完成时处理任务 - 但每个任务需要不同的方法来处理任务的结果
【发布时间】:2014-07-10 19:01:37
【问题描述】:

我正在使用 async/await 来调用一些外部 API。它们都返回一个字符串值,但格式不同,需要自己处理。 我想在任务完成时处理返回的值。我不想等到一切都完成,因此我正在使用 Task.WhenAny()。如何在任务完成时处理任务,并且在完成每个任务时仍然使用正确的“处理”方法?

我在第一次发帖后做了一些更改,这是我最新的:

public async Task<List<string>> Get()
{
    var task1 = Method1Async();
    var task2 = Method1Async();
    var tasks = new List<Task<string>> {task1, task2};
    var results = new List<string>();
    while (tasks.Count > 0)
    {
        var justCompletedTask = await Task.WhenAny(tasks);//will not throw
        tasks.Remove(justCompletedTask);
        try
        {
           var result = await justCompletedTask;
           results.Add(result);
        }
        catch(Exception)
        { 
          //deal with it
        }
    }

    return results;
}

private async Task<string> Method1Async()
{
    //this may throw - something like forbidden or any other exception
    var task = _httpClient.GetStringAsync("api1's url here");
    var result = await Method1ResultProcessorAsync(task);
    return result;
}

private async Task<string> Method1ResultProcessorAsync(Task<string> task)
{
    //process task's result -if it successuflly completed and return that
    return await task; //for now
}

private async Task<string> Method2Async()
{
    //this may throw - something like forbidden or any other exception
    var task = _httpClient.GetStringAsync("api2's url here");
    var result = await Method2ResultProcessorAsync(task);
    return await task;
}

private async Task<string> Method2ResultProcessorAsync(Task<string> task)
{
    //This processing logic is entirely different from Method1ResultProcessor
    //process task's result -if it successfully completed and return that 
    return await task; //for now
}

我有两个问题:

  1. 这是解决问题的正确方法吗?
  2. 这里如何更好地处理异常?这是非常重要的,所以一个人的失败不应该让整个事情失败。只要任何一种方法成功,就可以了。但是如果都失败了,我想把Get方法扔掉。

【问题讨论】:

  • 显示你正在使用的代码。
  • 我用代码示例更新了它
  • 能否请您使用您使用的语言进行标记。我以前从未见过 :-)
  • 它是 C# 5.0 的异步/等待特性。已添加标签。感谢 I3arnon 和 Klaus 提出的观点

标签: c# task-parallel-library async-await c#-5.0


【解决方案1】:

由于您的处理器方法已经接受Task,您可以直接调用它们,它们将异步等待相应的结果:

public Task<string[]> Get()
{
    var task1 = Method1ResultProcessorAsync(Method1Async());
    var task2 = Method2ResultProcessorAsync(Method2Async());

    return Task.WhenAll(task1, task2);
}

按照您描述的方式处理异常会使事情变得更加复杂,但您可以使用类似的方法:

public async Task<List<string>> Get()
{
    var task1 = Method1ResultProcessorAsync(Method1Async());
    var task2 = Method2ResultProcessorAsync(Method2Async());

    var tasks = new[] { task1, task2 };

    try
    {
        await Task.WhenAll(tasks);
    }
    catch {}

    var results = tasks.Where(t => t.Status == TaskStatus.RanToCompletion)
                       .Select(t => t.Result)
                       .ToList();

    if (results.Any())
        return results;

    // or maybe another exception,
    // since await handles AggregateException in a weird way
    throw new AggregateException(tasks.Select(t => t.Exception));
}

【讨论】:

  • 我想我更喜欢这个——我想它会更易读。但是我会将结果处理器作为参数传递给外部 api 调用方法,或者使用 continuewith 链接来代替下面提到的@modosansreves。它需要一些清理,但我认为这回答了问题。
【解决方案2】:
  1. 这是描述Method1Async()Method2Async() 的另一种方式。这是ContinueWith的演示。回答您在标题中提出的问题 - 您确实可以为每个任务使用不同的方法,该方法将在任务完成后调用

    var task1 = _httpClient.GetStringAsync("api1's url here").ContinueWith(t => Method1ResultProcessorAsync(t));
    var task2 = _httpClient.GetStringAsync("api2's url here").ContinueWith(t => Method2ResultProcessorAsync(t));
    
  2. 您正确处理了异常。回答“但如果都失败了,我想用Get方法抛出。”:只需检查results.Count == 0是否在return之前,如果是0则throw

【讨论】:

  • 感谢您的帖子。但这与我所拥有的有何不同?你现在有两个任务(task1 和 task2),你什么时候检查所有任务(在这种情况下是 2 个任务)是否都失败了?能否请您发布您的解决方案。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-11-12
  • 2019-06-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多