Вернуть результаты заданий - включая исключения? - PullRequest
0 голосов
/ 12 февраля 2019

Следующий код запускает M1, M2, M3, M4 параллельно.Каждый метод может вызывать исключения.Метод должен возвращать результаты четырех асинхронных методов - либо int, возвращаемого методами, либо исключения.

async Task<string> RunAll()
{
    int m1result, m2result, m3result, m4result;
    try
    {
        var m1task = M1();
        var m2task = M2();
        var m3task = M3();
        var m4task = M4();
        // await Task.WhenAll(new Task<int>[] { m1task, m2task, m3task, m4task });
        m1result = await m1task;
        m2result = await m2task;
        m3result = await m3task;
        m4result = await m4task;
    }
    catch (Exception ex)
    {
        // need to return the ex of the failed task. How?
    }
    // How to implement M1HasException, M2HasException, ... in the following lines?
    var m1msg = M1HasException ? M1ExceptionMessage : m1result.ToString();
    var m2msg = M2HasException ? M2ExceptionMessage : m2result.ToString();
    var m3msg = M3HasException ? M3ExceptionMessage : m3result.ToString();
    var m4msg = M4HasException ? M4ExceptionMessage : m4result.ToString();
    return $"M1: {m1msg}, M2: {m2msg}, M3: {m3msg}, M4: {m4msg}";
}

Как записать отдельные исключения из невыполненной задачи?

Например, если только M2 сгенерировал исключение,

"M1: 1, M2: Excpetion...., M3: 3, M4: 4"

Ответы [ 4 ]

0 голосов
/ 12 февраля 2019

Вы можете обернуть задачи в WaitAll и поймать AggregateException ( документы ),

try
{
    Task.WaitAll(new[] { task1, task2 }, token);
}
catch (AggregateException ae)
{
    foreach (var ex in ae.InnerExceptions)
        //Do what ever you want with the ex.
}
0 голосов
/ 12 февраля 2019

Не могли бы вы обернуть каждый await в try-catch блок и перехватить сообщение об исключении, если оно есть, кажется возможным ...

var results = new List<string>();

try { results.Add(await t1); } catch { results.Add("Exception"); };
try { results.Add(await t2); } catch { results.Add("Exception"); };
try { results.Add(await t3); } catch { results.Add("Exception"); };

return string.Join("|", results);

, если вы хотите использовать WhenAll, вы можете awaitдля этого и игнорируйте исключения, а затем выполните то же упражнение, как показано выше, чтобы получить результаты отдельных задач ...

try { await Task.WhenAll(t1, t2, t3); } catch { };
//                                      ^^^^^^^^^
// then same as ^ above
0 голосов
/ 12 февраля 2019

Как указывалось в других ответах / комментариях, одним из возможных подходов является использование ContinueWith или ContinueWhenAll.Это хитрый трюк, потому что Task имеет свойство Exception:

Получает исключение AggregateException, которое привело к преждевременному завершению задачи.Если Задача успешно выполнена или еще не выдало никаких исключений, возвращается нулевое значение.

При использовании ContinueWith независимо от того, успешно ли задача завершается или нет, она будет передана в качестве аргумента функции делегата.,Оттуда вы можете проверить, было ли выдано исключение.

Task<string> GetStringedResult<T>(Task<T> initialTask)
{
    return initialTask.ContinueWith(t => {
        return t.Exception?.InnerException.Message ?? t.Result.ToString();
    });
}

async Task<string> RunAll()
{
    string m1result, m2result, m3result, m4result;

    var m1task = GetStringedResult(M1());
    var m2task = GetStringedResult(M2());
    var m3task = GetStringedResult(M3());
    var m4task = GetStringedResult(M4());

    m1result = await m1task;
    m2result = await m2task;
    m3result = await m3task;
    m4result = await m4task;

    return $"M1: {m1result}, M2: {m2result}, M3: {m3result}, M4: {m4result}";
}
0 голосов
/ 12 февраля 2019

Каждая задача имеет свойство «Статус» и «Исключение».

Возможно, вы захотите узнать, произошла ли ошибка:

myTask.Status == TaskStatus.Faulted

Или, если она произошла:

if (myTask.Exception != null) 

Вы можете использовать ContinueWhenAll для запуска всехзадачи, а затем проверить статус.

См. Документы здесь .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...