.NET 并行(多核)编程系列之六 Task基础部分完结篇

简介: 原文:.NET 并行(多核)编程系列之六 Task基础部分完结篇.NET 并行(多核)编程系列之六 Task基础部分完结篇   前言:之前的文章介绍了了并行编程的一些基本的,也注重的讲述了Task的一些使用方法,本篇很短,将会结束Task的基础知识的介绍。
原文: .NET 并行(多核)编程系列之六 Task基础部分完结篇

.NET 并行(多核)编程系列之六 Task基础部分完结篇

  前言:之前的文章介绍了了并行编程的一些基本的,也注重的讲述了Task的一些使用方法,本篇很短,将会结束Task的基础知识的介绍。

 

  本篇的主要议题如下:

  1.       获取Task的状态

  2.       执行晚加载的Task(Lazily Task)

  3.       常见问题的解决方案

 

  系列文章链接:

  .NET 4 并行(多核)编程系列之一入门介绍

  .NET 4 并行(多核)编程系列之二 从Task开始 

  .NET 4 并行(多核)编程系列之三 从Task的取消 

  .NET 4 并行(多核)编程系列之四 Task的休眠 

  .NET 并行(多核)编程系列之五 Task执行和异常处理 

  .NET 并行(多核)编程系列之六 Task基础部分完结篇 

  .NET 并行(多核)编程系列之七 共享数据问题和解决概述

 

  1.       获取Task的状态

        在.NET并行编程还有一个已经标准化的操作就是可以获取task的状态,通过Task.Status属性来得到的,这个属性返回一个System.Threading.Tasks.TaskStatus的枚举值。

如下:

       Created:表明task已经被初始化了,但是还没有加入到Scheduler中。

       WatingForActivationtask正在等待被加入到Scheduler中。

 WaitingToRun:已经被加入到了Scheduler,等待执行。

       Runningtask正在运行

       WaitingForChildrenToComplete:表明父task正在等待子task运行结束。

       RanToCompletion:表明task已经执行完了,但是还没有被cancel,而且也这个task也没有抛出异常。

       Canceled:表明task已经被cancel了。(大家可以参看之前讲述取消task的文章)

       Faulted:表明task在运行的时候已经抛出了异常。

 

  2.       执行晚加载的Task(Lazily Task)

晚加载,或者又名延迟初始化,主要的好处就是避免不必要的系统开销。在并行编程中,可以联合使用Lazy变量和Task<>.Factory.StartNew()做到这点。(Lazy变量时.NET 4中的一个新特性,这里大家不用知道Lazy的具体细节)

       Lazy变量只有在用到的时候才会被初始化。所以我们可以把Lazy变量和task的创建结合:只有这个task要被执行的时候才去初始化。

       下面还是通过例子来讲解: 

 

img_405b18b4b6584ae338e0f6ecaf736533.gif 代码
   static   void  Main( string [] args)
        {
            
//  define the function
            Func < string >  taskBody  =   new  Func < string > (()  =>
            {
                Console.WriteLine(
" Task body working... " );
                
return   " Task Result " ;
            });

            
//  create the lazy variable
            Lazy < Task < string >>  lazyData  =   new  Lazy < Task < string >> (()  =>
            Task
< string > .Factory.StartNew(taskBody));

            Console.WriteLine(
" Calling lazy variable " );
            Console.WriteLine(
" Result from task: {0} " , lazyData.Value.Result);

            
//  do the same thing in a single statement
            Lazy < Task < string >>  lazyData2  =   new  Lazy < Task < string >> (
            () 
=>  Task < string > .Factory.StartNew(()  =>
            {
                Console.WriteLine(
" Task body working... " );
                
return   " Task Result " ;
            }));

            Console.WriteLine(
" Calling second lazy variable " );
            Console.WriteLine(
" Result from task: {0} " , lazyData2.Value.Result);

            
//  wait for input before exiting
            Console.WriteLine( " Main method complete. Press enter to finish. " );
            Console.ReadLine();
        }

 

      

  首先我们回想一下,在之前的系列文章中我们是怎么定义一个task的:直接new,或者通过taskfactory来创建,因为创建task的代码是在main函数中的,所以只要new了一个task,那么这个task就被初始化。现在如果用了Lazytask,那么现在我们初始化的就是那个Lazy变量了,而没有初始化task(初始化Lazy变量的开销小于初始化task),只有当调用了lazyData.Value时,Lazy变量中包含的那个task才会初始化。(这里欢迎大家提出自己的理解) 

 

  3.       常见问题的解决方案

a.       Task 死锁

描述:如果有两个或者多个task(简称TaskA)等待其他的taskTaskB)执行完成才开始执行,但是TaskB也在等待TaskA执行完成才开始执行,这样死锁就产生了。

 

解决方案:避免这个问题最好的方法就是:不要使的task来依赖其他的task。也就是说,最好不要你定义的task的执行体内包含其他的task

 

例子:在下面的例子中,有两个task,他们相互依赖:他们都要使用对方的执行结果。当主程序开始运行之后,两个task也开始运行,但是因为两个task已经死锁了,所以主程序就一直等待。

 

 

img_405b18b4b6584ae338e0f6ecaf736533.gif 代码
   static   void  Main( string [] args)
        {
            
//  define an array to hold the Tasks
            Task < int > [] tasks  =   new  Task < int > [ 2 ];

            
//  create and start the first task
            tasks[ 0 =  Task.Factory.StartNew(()  =>
            {
                
//  get the result of the other task,
                
//  add 100 to it and return it as the result
                 return  tasks[ 1 ].Result  +   100 ;
            });

            
//  create and start the second task
            tasks[ 1 =  Task.Factory.StartNew(()  =>
            {
                
//  get the result of the other task,
                
//  add 100 to it and return it as the result
                 return  tasks[ 1 ].Result  +   100 ;
            });


            
//  wait for the tasks to complete
            Task.WaitAll(tasks);

            
//  wait for input before exiting
            Console.WriteLine( " Main method complete. Press enter to finish. " );
            Console.ReadLine();
        }

 

 

   本篇就到这里了,很短,基础的部分就基本介绍完了,后面的文章就开始讲述应用。

   版权为小洋和博客园所有,转载请标明出处给作者。

   http://www.cnblogs.com/yanyangtian

 

 

 

 

 

 

 

 

 

 

 

 

 

 

目录
相关文章
|
存储 JSON 开发工具
Visual Studio编程效率提升技巧集(提高.NET编程效率)
Visual Studio编程效率提升技巧集(提高.NET编程效率)
215 0
Visual Studio编程效率提升技巧集(提高.NET编程效率)
|
11月前
|
传感器 数据采集 物联网
探索.NET nanoFramework:为嵌入式设备编程的新途径
探索.NET nanoFramework:为嵌入式设备编程的新途
413 7
|
11月前
.NET 4.0下实现.NET4.5的Task类相似功能组件
【10月更文挑战第29天】在.NET 4.0 环境下,可以使用 `BackgroundWorker` 类来实现类似于 .NET 4.5 中 `Task` 类的功能。`BackgroundWorker` 允许在后台执行耗时操作,同时不会阻塞用户界面线程,并支持进度报告和取消操作。尽管它有一些局限性,如复杂的事件处理模型和不灵活的任务管理方式,但在某些情况下仍能有效替代 `Task` 类。
144 0
|
大数据 开发工具 开发者
从零到英雄:.NET核心技术带你踏上编程之旅,构建首个应用,开启你的数字世界探险!
【8月更文挑战第28天】本文带领读者从零开始,使用强大的.NET平台搭建首个控制台应用。无论你是新手还是希望扩展技能的开发者,都能通过本文逐步掌握.NET的核心技术。从环境搭建到创建项目,再到编写和运行代码,详细步骤助你轻松上手。通过计算两数之和的小项目,你不仅能快速入门,还能为未来开发更复杂的应用奠定基础。希望本文为你的.NET学习之旅开启新篇章!
98 1
|
存储 C#
揭秘C#.Net编程秘宝:结构体类型Struct,让你的数据结构秒变高效战斗机,编程界的新星就是你!
【8月更文挑战第4天】在C#编程中,结构体(`struct`)是一种整合多种数据类型的复合数据类型。与类不同,结构体是值类型,意味着数据被直接复制而非引用。这使其适合表示小型、固定的数据结构如点坐标。结构体默认私有成员且不可变,除非明确指定。通过`struct`关键字定义,可以包含字段、构造函数及方法。例如,定义一个表示二维点的结构体,并实现计算距离原点的方法。使用时如同普通类型,可通过实例化并调用其成员。设计时推荐保持结构体不可变以避免副作用,并注意装箱拆箱可能导致的性能影响。掌握结构体有助于构建高效的应用程序。
422 7
|
Java Spring 自然语言处理
Spring 框架里竟藏着神秘魔法?国际化与本地化的奇妙之旅等你来揭开谜底!
【8月更文挑战第31天】在软件开发中,国际化(I18N)与本地化(L10N)对于满足不同地区用户需求至关重要。Spring框架提供了强大支持,利用资源文件和`MessageSource`实现多语言文本管理。通过配置日期格式和货币符号,进一步完善本地化功能。合理应用这些特性,可显著提升应用的多地区适应性和用户体验。
129 0
|
传感器 数据采集 物联网
探索未来:.NET nanoFramework引领嵌入式设备编程革新之旅
【8月更文挑战第28天】.NET nanoFramework 是一款专为资源受限的嵌入式设备设计的轻量级、高性能框架,基于 .NET Core,采用 C# 进行开发,简化了传统底层硬件操作的复杂性,极大提升了开发效率。开发者可通过 Visual Studio 或 Visual Studio Code 快速搭建环境并创建项目,利用丰富的库和驱动程序轻松实现从基础 LED 控制到网络通信等多种功能,显著降低了嵌入式开发的门槛。
242 0
|
JSON 编解码 Go
Golang深入浅出之-HTTP客户端编程:使用net/http包发起请求
【4月更文挑战第25天】Go语言`net/http`包提供HTTP客户端和服务器功能,简化高性能网络应用开发。本文探讨如何发起HTTP请求,常见问题及解决策略。示例展示GET和POST请求的实现。注意响应体关闭、错误处理、内容类型设置、超时管理和并发控制。最佳实践包括重用`http.Client`,使用`context.Context`,处理JSON以及记录错误日志。通过实践这些技巧,提升HTTP编程技能。
177 1
|
Go 开发者
Golang深入浅出之-HTTP客户端编程:使用net/http包发起请求
【4月更文挑战第24天】Go语言的`net/http`包在HTTP客户端编程中扮演重要角色,但使用时需注意几个常见问题:1) 检查HTTP状态码以确保请求成功;2) 记得关闭响应体以防止资源泄漏;3) 设置超时限制,避免长时间等待;4) 根据需求处理重定向。理解这些细节能提升HTTP客户端编程的效率和质量。
170 1
|
SQL 开发框架 安全
【.NET Core】深入理解任务并行库 (TPL)
【.NET Core】深入理解任务并行库 (TPL)
180 0