【问题标题】:Regarding usage of Task.Start() , Task.Run() and Task.Factory.StartNew()关于 Task.Start() 、 Task.Run() 和 Task.Factory.StartNew() 的使用
【发布时间】:2015-06-23 23:02:34
【问题描述】:

我刚刚看到了 3 个关于 TPL 使用的例程,它们做同样的工作;这是代码:

public static void Main()
{
    Thread.CurrentThread.Name = "Main";

    // Create a task and supply a user delegate by using a lambda expression. 
    Task taskA = new Task( () => Console.WriteLine("Hello from taskA."));
    // Start the task.
    taskA.Start();

    // Output a message from the calling thread.
    Console.WriteLine("Hello from thread '{0}'.", 
                  Thread.CurrentThread.Name);
    taskA.Wait();
}

public static void Main()
{
    Thread.CurrentThread.Name = "Main";

    // Define and run the task.
    Task taskA = Task.Run( () => Console.WriteLine("Hello from taskA."));

    // Output a message from the calling thread.
    Console.WriteLine("Hello from thread '{0}'.", 
                      Thread.CurrentThread.Name);
    taskA.Wait();
}

public static void Main()
{
    Thread.CurrentThread.Name = "Main";

    // Better: Create and start the task in one operation. 
    Task taskA = Task.Factory.StartNew(() => Console.WriteLine("Hello from taskA."));

    // Output a message from the calling thread.
    Console.WriteLine("Hello from thread '{0}'.", 
                    Thread.CurrentThread.Name);

    taskA.Wait();                  
}

我只是不明白为什么 MS 提供了 3 种不同的方式在 TPL 中运行作业,因为它们都工作相同:Task.Start()Task.Run()Task.Factory.StartNew()

告诉我,Task.Start()Task.Run()Task.Factory.StartNew() 是否都用于相同的目的,或者它们有不同的意义?

什么时候应该使用Task.Start(),什么时候应该使用Task.Run(),什么时候应该使用Task.Factory.StartNew()

请通过示例帮助我根据场景详细了解它们的实际用法,谢谢。

【问题讨论】:

标签: c# .net async-await task-parallel-library


【解决方案1】:

Task.Run 是带有特定安全参数的Task.Factory.StartNew 的简写:

Task.Factory.StartNew(
    action, 
    CancellationToken.None, 
    TaskCreationOptions.DenyChildAttach, 
    TaskScheduler.Default);

它是在 .Net 4.5 中添加的,以帮助越来越频繁地使用 async 并将工作卸载到 ThreadPool

Task.Factory.StartNew(在 .Net 4.0 中添加了 TPL)更加健壮。仅当 Task.Run 不够时才应使用它,例如当您想使用 TaskCreationOptions.LongRunning 时(尽管当委托是异步的时它是不必要的。更多关于我的博客:LongRunning Is Useless For Task.Run With async-await)。更多关于Task.Factory.StartNewTask.Run vs Task.Factory.StartNew

永远不要创建Task 并调用Start(),除非您找到非常好的理由这样做。仅当您有一些需要创建任务但不安排它们的部分和另一个需要安排但不创建的部分时才应该使用它。这几乎不是一个合适的解决方案,而且可能很危险。更多内容"Task.Factory.StartNew" vs "new Task(...).Start"

总之,主要使用Task.Run,如果必须使用Task.Factory.StartNew,永远不要使用Start

【讨论】:

【解决方案2】:

简答

如果您不使用嵌套子任务并且总是希望您的任务在线程池上执行,最好使用Task.Run

长答案:

Task.RunTask.Factory.StartNew 都支持创建和调度 Task 对象,因此我们不需要创建 Task 并调用 Start()

Task.Run(action);

相当于:

Task.Factory.StartNew(action, CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default);

Task.Factory.StartNew(action);

相当于:

Task.Factory.StartNew(action, CancellationToken.None, TaskCreationOptions.None, TaskScheduler.Current);

Task.Run 使用TaskCreationOptions.DenyChildAttach 表示子任务不能附加到父任务,它使用TaskScheduler.Default 表示在线程池上运行任务的将始终用于运行任务。

Task.Factory.StartNew 使用TaskScheduler.Current 表示当前线程的调度程序,它可能是TaskScheduler.Default,但并非总是如此。

【讨论】:

    猜你喜欢
    • 2013-06-04
    • 2016-11-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-01-21
    相关资源
    最近更新 更多