什么是TaskFactory.StartNew()中的“cancellationToken”用于?

时间:2010-09-11 04:09:28

标签: c# multithreading cancellation-token

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,这是无用的,因为我仍然需要手动实现。

2 个答案:

答案 0 :(得分:34)

由于评论,我正在发布另一个答案。

请考虑以下代码:

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中指定令牌参数时,任务将立即被取消,而不会分配工作线程。

答案 1 :(得分:3)

使用任务取消仍然是合作的。你不希望在一些关键操作中杀死一个线程。你需要检查它。

CancellationTokens比简单的ManualResetEvent构造更好地用于信号关闭操作,因为你可以级联或组合它们,例如,你可以有一个用于整个应用程序关闭,你可以将它与一个用于取消一项特殊的任务。该任务只需查看一个CancellationToken,但您可以从CancellationTokenSource取消该任务。