8

キャンセル トークンを使用して (タイムアウトが終了する前に) タイムアウトのあるタスクをキャンセルすると、例外がスローされます。例:

mytask.start();
bool didTaskRunInTime = mytask.wait(5 mins, _cancelToken);

つまり、以下のように進むことはできません。

//was the task cancelled
if (_cancelToken.IsCancelRequested)
{
    // log cancel from user to file etc
}

if (didTaskRunInTime )
{
    int taskResult = myTask.Result;
    // log result to file
}
else if (!_cancelToken.IsCancelRequested)
{
    // Tell user task timed out , log a message etc
}

これらすべてを catch ブロックで行う必要があり、コードが乱雑に見えます。これを行う正しい方法は何ですか?

4

2 に答える 2

17

そのタスクだけTask.WaitAnyの配列で呼び出すことができます。その後、タスクのステータスに基づいてアクションを実行できますが、メソッドは戻ります。サンプルコード:

using System;
using System.Threading;
using System.Threading.Tasks;

class Test
{
    static void Main()
    {
        Task sleeper = Task.Factory.StartNew(() => Thread.Sleep(100000));

        int index = Task.WaitAny(new[] { sleeper },
                                 TimeSpan.FromSeconds(0.5));
        Console.WriteLine(index); // Prints -1, timeout

        var cts = new CancellationTokenSource();

        // Just a simple wait of getting a cancellable task
        Task cancellable = sleeper.ContinueWith(ignored => {}, cts.Token);

        // It doesn't matter that we cancel before the wait
        cts.Cancel();

        index = Task.WaitAny(new[] { cancellable },
                             TimeSpan.FromSeconds(0.5));
        Console.WriteLine(index); // 0 - task 0  has completed (ish :)
        Console.WriteLine(cancellable.Status); // Cancelled
    }
}

タスクが失敗した場合は、ファイナライズされたときに例外が発生しないように、例外を「観察」する必要があることに注意してください:)

于 2011-07-13T16:26:16.313 に答える
3

OperationCanceledException を使用してみてください

try
{           
    mytask.start();
    bool didTaskRunInTime = mytask.wait(5 mins, _cancelToken);

    if (didTaskRunInTime )
    {
        int taskResult = myTask.Result;
        //log result to file
    }
    else
    {
        // Tell user task timed out , log a message etc
    }
}
catch (OperationCanceledException ex)
{
    // log cancel from user to file et
}
于 2011-07-13T16:29:05.623 に答える