多核时代 .NET Framework 4 中的并行编程3---任务并行库之Task (下)

简介:

1.     创建可取消操作的Task

一个任务Task开始之后,我们可以根据需要对任务进行取消,以便停止当前执行的操作.代码如下:

   static void Main(string[] args)

        {

           

            CancellationTokenSource s = new CancellationTokenSource();

            CancellationToken token = s.Token;

            Task task = new Task(() =>

                {

                    for (int i = 0; i < int.MaxValue; i++)

                    {

                     

                        if (token.IsCancellationRequested)

                        {

                            Console.WriteLine("任务被取消");

                            throw  new OperationCanceledException(token);

                        }

                        else

                        {

                            Console.WriteLine("当前值 {0}", i);

                        }

                    }

                }, token);

 

            token.Register(() =>

            {

                Console.WriteLine("任务取消时执行的委托方法");

            });

 

 

           

            task.Start();

            Console.ReadLine();

 

            Console.WriteLine("取消任务.");

            s.Cancel();

 

            Console.WriteLine("Main执行完毕.");

            Console.ReadLine();

        }

 

从上面的代码示例中,我们可以看到创建可取消任务Task的过程是:先创建一个可取消操作的源CancellationTokenSource的一个对象,然后使用CancellationTokenSource的可取消操作结构对象CancellationToken,并使用该结构初始化一个任务Task,最后通过调用可取消操作的源的Cancel()方法,来取消当前关联任务的操作.

简单的理解,就是将一个任务Task与可取消操作的源CancellationTokenSource通过CancellationToken关联起来通过CancellationTokenSource的取消方法来通知相应的任务取消操作.(可以使用某个CancellationToken初始化多个任务Task,然后在调用CancellationTokenSource的取消方法时,则该多个任务也将被通知取消,从而到达一次取消多个任务的目的)。

在上面的代码,可以看到:

Ø token.IsCancellationRequested,它是获取是否已请求取消此标记, 如果此属性为 true,则只能保证已请求取消。 它不保证每个注册的处理程序已完成执行,也不能保证取消请求已完成传播到所有注册的处理程序.也就是说,此属性只是可以得到一个任务ask获取到了取消标记,但它不能代表此任务已经停止完毕或执行完毕取消时注册的事件(token.Register)。

Ø token.Register(…)向该任务注册一个取消的委托,在任务被取消时,将会执行此委托。

通过上面的代码,我们应该学会了如何创建一个可以被取消的任务Task,如何判断一个任务Task已经被通知被取消,如何给一个任务Task注册一个取消的处理程序。此外,Task类的IsCanceled属性可以执行任务是否被取消执行完毕。

 

2.     Task的等待

任务Task可以在某些时候去等待某个事件发生或者等待其他任务执行完毕,Task提供了:

Ø Wait(等待某个事件发生,这就好比:请客吃饭,等一个朋友到来我们就吃饭。)。

Ø WaitAll(等待某组事件中全部事件都已发生,这就好比:请客吃饭,必须等所有受邀的朋友都到齐才能吃饭)。

Ø WaitAny(等待某组事件中任何一个事件都已发生,这就好比:请客吃饭,只要所有受邀的朋友中有任何一个朋友,不论是谁,只要有朋友到了,我们就开始吃饭)。

相应的,这些方法也有很多重载。

下面来看看具体的代码:

        static void WaitTask()

        {

            CancellationTokenSource ts = newCancellationTokenSource();

            CancellationToken token = ts.Token;

            Task task1 = new Task(() =>

            {

 

                for (int i = 0; i < 10; i++)

                {

                    token.ThrowIfCancellationRequested();

                    Console.WriteLine("任务1中i的值是{0}", i);

                    token.WaitHandle.WaitOne(1 * 1000);

                }

                Console.WriteLine("任务1执行完毕");

            }, token);

           

            Console.WriteLine("任务当前状态:{0}", task1.Status.ToString());

 

            Task task2 = new Task(() =>

            {

                Console.WriteLine("任务2执行完毕");

            }, token);

 

            task1.Start();

            task2.Start();

            Console.WriteLine("任务当前状态:{0}", task1.Status.ToString());

 

            Console.WriteLine("等待所有任务执行完毕…..");

            Task.WaitAll(task1, task2);

    Console.WriteLine("任务当前状态:{0}", task1.Status.ToString());

 

            Console.WriteLine("所有任务全部执行完毕.");

            Console.ReadLine();

        }

需要说明的是:

Ø 代码token.ThrowIfCancellationRequested();与if(token.IsCancellationRequested) throw new OperationCanceledException(token);是完全等效,只是不同的写法而已。

Ø 代码token.WaitHandle.WaitOne1*1000)是当前执行该代码的任务Task休息1秒中,类似Threed.Sleep(1*1000)。

Ø Task.WaitAlltask1,task2….)是等待所有的任务。

Ø Task Status 获取此任务的 TaskStatus。

其中TaskStatus枚举有:

(1)     Created 该任务已初始化,但尚未被计划。 

(2)     WaitingForActivation 该任务正在等待 .NET Framework 基础结构在内部将其激活并进行计划。 

(3)     WaitingToRun 该任务已被计划执行,但尚未开始执行。 

(4)     Running 该任务正在运行,但尚未完成。 

(5)     WaitingForChildrenToComplete 该任务已完成执行,正在隐式等待附加的子任务完成。 

(6)     RanToCompletion 已成功完成执行的任务。 

(7)     Canceled 该任务已通过对其自身的 CancellationToken 引发 OperationCanceledException 对取消进行了确认,此时该标记处于已发送信号状态;或者在该任务开始执行之前,已向该任务的 CancellationToken 发出了信号。 

(8)     Faulted 由于未处理异常的原因而完成的任务。 

 此外,Task还有以下几个重要属性

Ø IsCanceled 获取此 Task 实例是否由于被取消的原因而已完成执行。 

Ø IsCompleted 获取此 Task 是否已完成。 

Ø IsFaulted 获取 Task 是否由于未经处理异常的原因而完成。 

    好,本节介绍完毕。

 



    本文转自风车车  博客园博客,原文链接:http://www.cnblogs.com/xray2005/archive/2011/08/22/2148815.html ,如需转载请自行联系原作者

相关文章
|
1月前
|
C#
一个.NET开源、轻量级的运行耗时统计库 - MethodTimer
一个.NET开源、轻量级的运行耗时统计库 - MethodTimer
|
1月前
|
人工智能 自然语言处理 API
适用于 .NET 稳定的官方OpenAI库
适用于 .NET 稳定的官方OpenAI库
|
1月前
|
传感器 人工智能 供应链
.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。
本文深入探讨了.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。通过企业级应用、Web应用及移动应用的创新案例,展示了.NET在各领域的广泛应用和巨大潜力。展望未来,.NET将与新兴技术深度融合,拓展跨平台开发,推动云原生应用发展,持续创新。
32 4
|
2月前
|
安全 Java 网络安全
Android远程连接和登录FTPS服务代码(commons.net库)
Android远程连接和登录FTPS服务代码(commons.net库)
32 1
|
2月前
|
开发框架 缓存 监控
NET Framework 到 .NET 5/6 的迁移是重大的升级
本文详细介绍了从 .NET Framework 4.8 迁移到 .NET 5/6 的过程,通过具体案例分析了迁移策略与最佳实践,包括技术栈评估、代码迁移、依赖项更新及数据库访问层的调整,强调了分阶段迁移、保持代码可维护性及性能监控的重要性。
61 3
|
1月前
|
开发框架 安全 .NET
.NET使用Moq开源模拟库简化单元测试
.NET使用Moq开源模拟库简化单元测试~
|
2月前
|
网络协议 Unix Linux
一个.NET开源、快速、低延迟的异步套接字服务器和客户端库
一个.NET开源、快速、低延迟的异步套接字服务器和客户端库
|
2月前
.NET 4.0下实现.NET4.5的Task类相似功能组件
【10月更文挑战第29天】在.NET 4.0 环境下,可以使用 `BackgroundWorker` 类来实现类似于 .NET 4.5 中 `Task` 类的功能。`BackgroundWorker` 允许在后台执行耗时操作,同时不会阻塞用户界面线程,并支持进度报告和取消操作。尽管它有一些局限性,如复杂的事件处理模型和不灵活的任务管理方式,但在某些情况下仍能有效替代 `Task` 类。
|
2月前
|
开发者 Windows
.NET 开源扁平化、美观的 C/S 控件库
【10月更文挑战第23天】介绍了三款适用于 .NET 平台的开源扁平化、美观的 C/S 控件库:MaterialSkin 采用 Google Material Design 风格,适合现代感界面;Krypton Toolkit 提供丰富控件,界面易于定制;Fluent Ribbon Control Suite 模仿 Office 界面,适合复杂功能应用。每款控件库均附有示例代码及 GitHub 链接。
|
3月前
|
开发框架 前端开发 JavaScript
ASP.NET MVC 教程
ASP.NET 是一个使用 HTML、CSS、JavaScript 和服务器脚本创建网页和网站的开发框架。
48 7