Страницы

Поиск по вопросам

четверг, 9 января 2020 г.

Ограничить время жизни потока

#c_sharp #net #async_await


Есть список задач, каждую задачу нужно ограничить по времени. Если задача не успевает
выполниться, то пусть возвращается какое-то дефолтное значение.

class Program
{
    static void Main(string[] args)
    {
        var sw = new Stopwatch();
        sw.Start();

        foreach (var i in GetInts())
        {
            Console.WriteLine("result: " + i);
        }

        sw.Stop();
        Console.WriteLine("done " + sw.Elapsed.TotalMilliseconds.ToString("F2"));
        Console.ReadKey();
    }

    static List GetInts()
    {
        using (var cts = new CancellationTokenSource())
        {
            var tasks = new Task[3];

            tasks[1] = Task.Run(() =>
            {
                int i = 5000;
                Thread.Sleep(i);
                Console.WriteLine(i);
                return i;
            }, cts.Token);


            tasks[0] = Task.Run(() =>
            {
                int i = 3000;
                Thread.Sleep(i);
                Console.WriteLine(i);
                return i;
            }, cts.Token);

            tasks[2] = Task.Run(() =>
            {
                int i = 2000;
                Thread.Sleep(i);
                Console.WriteLine(i);
                return i;
            }, cts.Token);

            var delay = 1000;

            return tasks.Select(x => x.TimeoutAfter(delay, cts)).Select(x => x.Result).ToList();
        }
    }
}

public static class TaskExtensions
{
    public static async Task TimeoutAfter(this Task task,
int delay, CancellationTokenSource cts)
    {

        using (var timeoutCancellationTokenSource = new CancellationTokenSource())
        {

            var completedTask = await Task.WhenAny(task, Task.Delay(delay, timeoutCancellationTokenSource.Token));
            if (completedTask == task)
            {
                timeoutCancellationTokenSource.Cancel();
                return await task;
            }
            else
            {
                cts.Cancel();
                return default(TResult);
            }
        }
    }
}


Хотя в коде прописано ограничение в 1000мс, данный код выводит следующий результат:

2000
result: 0
result: 0
result: 2000
done 2010,83
3000
5000


Почему задача с 2с попадает в result? А если выставить delay = 3000, то выведутся
все задачи (хотя 5с не должна):

2000
3000
5000
result: 3000
result: 5000
result: 2000
done 5008,06

    


Ответы

Ответ 1



Смотрите. Вы не можете завершить бегущую задачу насильно, точно так же как вы не можете завершить насильно любой другой код. Для этого задача должна сотрудничать. Идея с использованием CancellationToken'а правильная, но не доведена до конца, отсюда и проблема. CancellationToken в Task.Run относится только к процессу запуска таска*. Когда таск запущен, он не сможет магическим образом узнать о том, что токен сработал, ему нужно самому проверять отмену токена. Например, так: var ct = cts.Token; tasks[0] = Task.Run(async () => // добавил async { int i = 3000; await Task.Delay(i, ct); // проверяем токен Console.WriteLine(i); return i; }, ct); Ещё по теме: Как работает CancellationToken в TaskFactory.StartNew Method (Action, CancellationToken)? *Почти. Ещё он ассоциируется с таском, и при выбросе исключения этим токеном таск переходит в состояние Cancelled, а не Faulted.

Комментариев нет:

Отправить комментарий