В методе MyTask() #2, подсчет равен 2
В методе MyTask() #1, подсчет равен 2
В методе MyTask() #2, подсчет равен 3
В методе MyTask() #1, подсчет равен 3
В методе MyTask() #2, подсчет равен 4
В методе MyTask() #1, подсчет равен 4
В методе MyTask() #2, подсчет равен 5
В методе MyTask() #1, подсчет равен 5
В методе MyTask() #2, подсчет равен 6
В методе MyTask() #1, подсчет равен 6
В методе MyTask() #2, подсчет равен 7
В методе MyTask() #1, подсчет равен 7
В методе MyTask() #2, подсчет равен 8
В методе MyTask() #1, подсчет равен 8
В методе MyTask() #2, подсчет равен 9
MyTask №2 завершен
В методе MyTask() #1, подсчет равен 9
MyTask №1 завершен
Основной поток завершен.
Как следует из приведенного выше результата, выполнение метода Main()
приостанавливается до тех пор, пока не завершатся обе задачи tsk
и tsk2
. Следует, однако, иметь в виду, что в рассматриваемой здесь программе последовательность завершения задач tsk
и tsk2
не имеет особого значения для вызовов метода Wait()
. Так, если первой завершается задача tsk2
, то в вызове метода tsk.Wait()
будет по-прежнему ожидаться завершение задачи tsk
. В таком случае вызов метода tsk2.Wait()
приведет к выполнению и немедленному возврату из него, поскольку задача tsk2
уже завершена.
В данном случае оказывается достаточно двух вызовов метода Wait()
, но того же результата можно добиться и более простым способом, воспользовавшись методом WaitAll()
. Этот метод организует ожидание завершения группы задач. Возврата из него не произойдет до тех пор, пока не завершатся все задачи. Ниже приведена простейшая форма объявления этого метода.
public static void WaitAll(params Task[] tasks)
Задачи, завершения которых требуется ожидать, передаются с помощью параметра в виде массива params
, то данному методу можно отдельно передать массив объектов типа Task
или список задач. При этом могут быть сгенерированы различные исключения, включая и AggregateException
.
Для того чтобы посмотреть, как метод WaitAll()
действует на практике, замените в приведенной выше программе следующую последовательность вызовов.
tsk.Wait();
tsk2.Wait();
на
Task.WaitAll(tsk, tsk2);
Программа будет работать точно так же, но логика ее выполнения станет более понятной.
Организуя ожидание завершения нескольких задач, следует быть особенно внимательным, чтобы избежать взаимоблокировок. Так, если две задачи ожидают завершения друг друга, то вызов метода WaitAll()
вообще не приведет к возврату из него. Разумеется, условия для взаимоблокировок возникают в результате ошибок программирования, которых следует избегать. Следовательно, если вызов метода WaitAll()
не приводит к возврату из него, то следует внимательно проанализировать, могут ли две задачи или больше взаимно блокироваться. (Вызов метода Wait()
, который не приводит к возврату из него, также может стать причиной взаимоблокировок.)
Иногда требуется организовать ожидание до тех пор, пока не завершится любая из группы задач. Для этой цели служит метод WaitAny()
. Ниже приведена простейшая форма его объявления.
public static int WaitAny(params Task[] tasks)
Задачи, завершения которых требуется ожидать, передаются с помощью параметра в виде массива Task
или отдельного списка аргументов типа Task
. Этот метод возвращает индекс задачи, которая завершается первой. При этом могут быть сгенерированы различные исключения.
Попробуйте применить метод WaitAny()
на практике, подставив в предыдущей программе следующий вызов.
Task.WaitAny(tsk, tsk2);
Теперь, выполнение метода Main()
возобновится, а программа завершится, как только завершится одна из двух задач.
Помимо рассматривавшихся здесь форм методов Wait(), WaitAll()
и WaitAny()
, имеются и другие их варианты, в которых можно указывать период простоя или отслеживать признак отмены. (Подробнее об отмене задач речь пойдет далее в этой главе.)
В классе Task
реализуется интерфейс IDisposable
, в котором определяется метод Dispose()
. Ниже приведена форма его объявления.