Что такое "cancelationToken" в приложении TaskFactory.StartNew()?
http://msdn.microsoft.com/en-us/library/dd988458.aspx
UPD
поэтому обсудим эту статью: http://msdn.microsoft.com/en-us/library/dd997396.aspx
Я немного изменил этот код:
static void Main()
{
var tokenSource2 = new CancellationTokenSource();
CancellationToken ct = tokenSource2.Token;
var task = Task.Factory.StartNew(() =>
{
// Were we already canceled?
ct.ThrowIfCancellationRequested();
bool moreToDo = true;
Thread.Sleep(5000);
while (moreToDo)
{
// Poll on this property if you have to do
// other cleanup before throwing.
if (ct.IsCancellationRequested)
{
Console.WriteLine("exit");
// Clean up here, then...
ct.ThrowIfCancellationRequested();
}
}
}, tokenSource2.Token); // this parameter useless
Console.WriteLine("sleep");
Thread.Sleep(2000);
Console.WriteLine("cancel");
tokenSource2.Cancel();
// Just continue on this thread, or Wait/WaitAll with try-catch:
try
{
task.Wait();
}
catch (AggregateException e)
{
foreach (var v in e.InnerExceptions)
{
Console.WriteLine(e.Message + " " + v.Message);
}
}
Console.ReadKey();
}
UPD: Ну, это меняет только task.IsCanceled
, что бесполезно, потому что я все еще должен выполнять все вручную.
Ответы
Ответ 1
Из-за комментариев, я отправляю еще один ответ.
Рассмотрим следующий код:
var tokenSource = new CancellationTokenSource();
CancellationToken ct = tokenSource.Token;
tokenSource.Cancel();
var task = Task.Factory.StartNew(() =>
{
// Were we already canceled?
ct.ThrowIfCancellationRequested();
// do some processing
});
Даже если вызов tokenSource.Cancel()
выдается до того, как задача была фактически запущена, вы по-прежнему выделяете рабочий поток из пула потоков, поэтому вы будете тратить некоторые системные ресурсы.
Но когда вы указываете аргумент токена в Task.Factory.StartNew
, задача будет немедленно отменена без выделения рабочего потока.
Ответ 2
Отмена с помощью заданий по-прежнему является совместной. Вы не хотите, чтобы поток был убит в середине некоторой критической операции. Вам нужно проверить его.
CancellationTokens лучше, чем простые конструкции, такие как ManualResetEvent
для сигнализации о завершении операции, потому что вы можете каскадировать или комбинировать их, например, вы можете иметь один для полного завершения работы приложения, и вы можете объединить его с одним для отмены конкретного задача. Задача должна только смотреть на CancellationToken
, но вы можете отменить ее либо из CancellationTokenSource
.