编写高质量代码改善C#程序的157个建议——建议82:Parallel简化但不等同于Task默认行为

建议82:Parallel简化但不等同于Task默认行为

建议81说到了Parallel的使用方法,不知道大家是否注意到文中使用的字眼:在同步状态下简化了Task的使用。也就是说,在运行Parallel中的For、ForEach方法时,调用者线程(在示例中就是主线程)是被阻滞的。Parallel虽然将任务交给Task去处理,即交给CLR线程池去处理,不过调用者会一直等到线程池中的相关工作全部完成。表示并行的静态类Parallel甚至只提供了Invoke方法,而没有同时提供一个BeginInvoke方法,这也从一定程度上说明了这个问题。

在使用Task时,我们最常使用的是Start方法(Task也提供了RunSynchronously),它不会阻滞调用者线程。如下所示:

static void Main()  
{  
    Task t = new Task(() =>
        {  
            while (true)  
            {  
 
            }  
        });  
    t.Start();  
    Console.WriteLine("主线程即将结束");  
    Console.ReadKey();  
} 

输出为:

主线程即将结束

使用Parallel执行相近的功能,主线程被阻滞:

static void Main()  
{  
  //在这里也可以使用Invoke方法  
    Parallel.For(0, 1, (i) =>
        {  
            while (true)  
            {  
 
            }  
        });  
    Console.WriteLine("主线程即将结束");  
    Console.ReadKey();  
} 

如果执行这段代码,永远不会有输出。

并行编程,意味着运行时在后台将任务分配到尽量多的CPU上,虽然它在后台使用Task进行管理,但这并不意味着它等同于异步。

转自:《编写高质量代码改善C#程序的157个建议》陆敏技