.Net4.0 任务(Task)[转]

.Net4.0 任务(Task)

任务(Task)是一个管理并行工作单元的轻量级对象。它通过使用CLR的线程池来避免启动专用线程,可以更有效率的利用线程池。System.Threading.Tasks 命名空间下任务相关类一览:

作用
Task 管理工作单元
Task<TResult> 管理带返回值的工作单元
TaskFactory 创建任务
TaskFactory<TResult> 创建任务或者有相同返回值的延续任务
TaskScheduler 管理任务调度
TaskCompletionSource 手动控制任务工作流

任务用来并行地执行工作,充分地利用多核:事实上,Parallel和PLINQ内部就是建立在任务并行的结构上。

任务提供了一系列强大的特性来管理工作单元,包括:

  • 协调任务调度
  • 建立一个任务从另一个任务中启动的父子关系
  • 实现合作取消(cooperative cancellation)模式
  • 无信号的任务等待
  • 附加延续任务(continuation)
  • 基于多个祖先任务调度一个延续任务
  • 传递异常到父任务、延续任务或任务消费者

同时任务实现了一个本地工作队列,它允许你高效地创建快速执行的子任务而不用遭受在单个工作队列时的竞争花费。任务并行库让你用最小的花费来创建成百上千的任务,但是如果你想创建上百万个任务,就必须分割这些任务到更大的工作单元,以保持效率。

创建与启动任务

有两种方法可以创建任务,一种是通过TaskFactory的StartNew()方法创建并启动任务;另一种是调用Task构造函数创建,然后手动启动任务。需要注意的是,任务启动后并不会立即执行,它是由任务调度器(TaskScheduler)来管理的。

  • TaskFactory的StartNew()方法创建任务的示例如下:

    //没有返回值
    Task.Factory.StartNew(() => Console.WriteLine("Task Created!"));
    //有返回值
    var task = Task.Factory.StartNew<string>(() => "Task Created!");
    Console.WriteLine(task.Result);

  • 调用Start方法手动启动的示例如下:

    var task = new Task<string>(() => "Task Created!");
    task.Start();//异步执行
    Console.WriteLine(task.Result);

  • 调用RunSynchronously方法手动启动的示例如下:

    var task = new Task<string>(() => "Task Created!");
    task.RunSynchronously();//同步执行
    Console.WriteLine(task.Result);

也可以在创建任务时指定一个任务状态参数,可以通过任务的AsyncState属性来访问该参数。示例:

var task = Task.Factory.StartNew(state => "hello " + state, "Mike");
Console.WriteLine(task.AsyncState);
Console.WriteLine(task.Result);

你还可以指定一个任务创建选项(TaskCreationOptions) ,这个枚举类型有以下枚举值:None,LongRunning,PreferFairness,AttachedToParent。下面解释各个枚举值的作用。

  • LongRunning:顾名思义就是长时间运行的任务,此选项建议任务调度器分配一个专用的线程给任务。这样做的原因是:长时间运行的任务可能会阻塞任务队列,导致那些短小的任务一直得不到执行。LongRunning也适合那些阻塞的任务。
  • PreferFairness:公平第一,此选项建议任务调度器尽量按照任务的启动时间来调度任务。但是它通常可能不这样做,因为它使用本地工作偷取队列(local work-stealing queues)优化任务调度。这个优化对那些非常小的任务很有用。
  • AttachToParent:附加到父任务,此选项用来创建子任务。创建子任务示例:

    第一种方式:
    var parent = Task.Factory.StartNew(() =>
    {
    var nonChildTask = Task.Factory.StartNew(
    () => Console.WriteLine("I‘m not a child task.")
    );
    var childTask = Task.Factory.StartNew(
    () => Console.WriteLine("I‘m a child task."),
    TaskCreationOptions.AttachedToParent);
    });
    第二种方式:
    Task parent=new Task(()=>
    {
    DoStep1();
    });
    Task task2 = parent.ContinueWith ((PrevTask) =>
    {
    DoStep2();
    });
    parent.Start();

任务等待

任务可以通过Wait()成员方法或Result属性来等待任务完成。

当调用Result属性时,将会执行下列操作:

  1. 如果任务已结束,返回任务结果
  2. 如果任务已开始,等待任务结束
  3. 如果任务尚未开始执行,则在当前线程执行任务

Task.WaitAny()静态方法等待任何一个任务完成。示例:

var tasks = new Task[3];
for (int i = 0; i < tasks.Length; i++)
{
int taskIndex = i;
tasks[i] = Task.Factory.StartNew(() =>
{
int seed=Guid.NewGuid().GetHashCode();
int waitTime = new Random(seed).Next(10, 100);
Thread.Sleep(waitTime);
Console.WriteLine("Task{0} Finished", taskIndex);
});
}
Task.WaitAny(tasks);
Console.WriteLine("已有任务完成");

Task.WaitAll()静态方法等待所有任务完成。即使有任务抛出异常也不会终止等待,它会在所有任务完成之后抛出一个AggregateException异常,这个异常聚合了所有任务抛出的异常。示例:

var tasks = new Task[3];
for (int i = 0; i < tasks.Length; i++)
{
int taskIndex = i;
tasks[i] = Task.Factory.StartNew(() =>
{
int waitTime = new Random(Guid.NewGuid().GetHashCode()).Next(10, 100);
Thread.Sleep(waitTime);
Console.WriteLine("Task{0} Finished", taskIndex);
});
}
Task.WaitAll(tasks);
Console.WriteLine("所有任务完成");

异常处理

默认情况下任务未处理的异常会终止应用程序。需要指出的是任务中未处理的异常不会立即导致应用程序终止,异常要延迟到垃圾回收器回收任务并调用Finalize方法时才会终止程序。如果读取了任务的Exception属性,这个操作将阻止随后的应用程序终止。
当等待任务完成时,所有未处理的异常会传递到调用方。
Wait()方法超时的异常也必须处理,否则导致应用程序终止。
子任务中未处理的异常会冒泡传递到父任务;嵌套任务中的非子任务的异常不会传递到这个任务的上一层任务,需要单独处理,否则将导致应用程序终止。

var task = Task.Factory.StartNew(() =>
{
Task.Factory.StartNew(() => { throw null; }, TaskCreationOptions.AttachedToParent);
Task.Factory.StartNew(() => { throw null; }, TaskCreationOptions.AttachedToParent);
Task.Factory.StartNew(() => { throw null; }, TaskCreationOptions.AttachedToParent);
});
task.Wait();

TaskScheduler.UnobservedTaskException静态事件提供了最后一种手段处理所有未处理异常。通过处理这个事件,就不用终止应用程序,而用你自己的异常处理逻辑替代它。

取消任务

当创建任务时,可以传入一个取消令牌(CancelationToken)参数,这样就可以安全的取消任务。示例:

var source = new CancellationTokenSource();
var token = source.Token;
var task = Task.Factory.StartNew(() =>
{
Console.WriteLine("Task starting...");
while (true)
{
token.ThrowIfCancellationRequested();
Console.WriteLine("I‘m alive. {0}",DateTime.Now);
Thread.Sleep(1000);
}
},token);

Task.Factory.StartNew(() =>
{
Thread.Sleep(4500);
source.Cancel();
});

try
{
task.Wait();
Console.WriteLine("Task stopped.");
}
catch (AggregateException e)
{
if (e.InnerException is OperationCanceledException)
{
Console.WriteLine("Task canceled.");
}
else
{
Console.WriteLine("errors.");
}
}

通过调用CancellationTokenSource的Cancel()方法取消任务,这个并不会立即终止任务,一直延迟到任务下次检测是否取消时才通过抛出OperationCanceledException终止任务。

如果想通过直接抛出OperationCanceledException异常的方式取消任务,则需要在任务中传入CancelationToken参数,否则就不能将任务的状态为TaskStatus.Canceled并触发OnlyOnCanceled延续任务。

此外,取消令牌也可以传递到Wait和CancelAndWait方法中来取消等待。

时间: 2024-10-26 20:59:28

.Net4.0 任务(Task)[转]的相关文章

.Net4.0 任务(Task)

.Net4.0 任务(Task),.net4.0任务task 任务(Task)是一个管理并行工作单元的轻量级对象.它通过使用CLR的线程池来避免启动专用线程,可以更有效率的利用线程池.System.Threading.Tasks 命名空间下任务相关类一览: 类 作用 Task 管理工作单元 Task<TResult> 管理带返回值的工作单元 TaskFactory 创建任务 TaskFactory<TResult> 创建任务或者有相同返回值的延续任务 TaskScheduler 管

(译).NET4.X并行任务Task需要释放吗?

摘要:本博文解释在.NET 4.X中的Task使用完后为什么不应该调用Dispose().并且说明.NET4.5对.NET4.0的Task对象进行的部分改进:减轻Task对WaitHandle对象的依赖,并且增强在释放了Task后对其成员的可访问性. 我多次获得这样一个问题: “Task实现了IDisposable接口并且公开Dispose()方法,这是否意味着我们要对所有的任务进行释放吗?” 概述 这是我对该问题的简要回答: “不是,不用释放你持有的Task.” 这是我对该问题的中篇回答: “

.Net4.0如何实现.NET4.5中的Task.Run及Task.Delay方法

前言 .NET4.0下是没有Task.Run及Task.Delay方法的,而.NET4.5已经实现,对于还在使用.NET4.0的同学来说,如何在.NET4.0下实现这两个方法呢? 在.NET4.0下,有一个泛型类,叫TaskCompletionSource<TReuslt>,它能控制Task的行为,如给Task设置结果.设置异常.设置取消等. MSDN是这样描述的(网址): 表示未绑定到委托的 Task<TResult> 的制造者方,并通过Task属性提供对使用者方的访问. 它有以

NET4.0多线程编程---Tasks

写在前面的话 不得不说4.0里面新增的task实在是让人耳目一新,曾经在thread里面查找当线程abort等等操作的时候有没有相应的事件,就像gridview在分页.绑定的时候会产生事件一样,但是在thread里面是没有找到.然而task里面这些都实现了,虽然不是以事件的方式实现,但是它真的实现了.很让人兴奋. 一.概述 在上一篇文章.NET4.0多线程编程---Cooperative Cancellation提到线程池没有提供任何内在的方法告诉我们操作什么时候完成,在线程完成了以后,我们也没

net4.0新特性之线程同步

有时候我们可能需要使用多线程来执行同一任务,这个任务可能包含多步,而每步之间可能并不相干,但是这个任务必须让所有步骤执行完成后才能够进入下一步.这就如同WF中的并行任务.在.net4.0之前我们可能需要几个类来做到同步.但是现在我们只需要1个类就OK. 代码 Console.WriteLine("任务启动"); using (CountdownEvent cd = new CountdownEvent(1)) { for (var i = 0; i < 5; i++) { cd.

.NET4.0多线程编程---Cooperative Cancellation

在多线程编程中线程池是不得不提的,在.net4.0之前一般对于线程池的说法是,把需要的线程交给线程池,我们自己将更多的关注点放在业务上,个人认为这既是线程池的优点也是缺点---加入线程池之后人为无法控制,也没有内在的方法告知线程的执行结果. 今天早晨拜读了Jeffrey的著作26th章.根据自己的理解今天主要学习“如何取消线程池中的线程”. 首先介绍System.Thread命名空间下面的两个类: CancellationToken ,其实这是一个结构,它主要传播有关应取消操作的通知. Canc

iis6下配置支持.net4.0&amp;发布网站[转]

iis6配置支持.net4.0 在win2003操作系统上发布两个网站,首先配置iis: 1.下载 .net framework 4.0   差不多48MB 2.安装 3.打开iis: 开始=>管理工具=>Internet 信息服务(IIS)管理器 4.点击管理器中的“web服务扩展”,刚刚安装的.net framework 4.0 就会出现在右边,选择并允许 这样就配置好了 有时还要同时配置.net framework 3.5和.net framework 4.0 才可以,那就再配置一次3.

连接Access数据库代码,以及.Net4.0版本的报表查询代码

using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using data1; using DA1; using data5; using DA5; namespace 连接仓库.管理员

类型:.net;问题:iis注册;结果:.net4.0注册到IIS ,重新注册IIS ,iis注册

.net4.0注册到IIS ,重新注册IIS ,iis注册 IIS和.netfw4.0安装顺序是从前到后,如果不小心颠倒了,无所谓. 打开程序-运行-cmd:输入一下命令重新注册IIS C:\WINDOWS\Microsoft.NET\Framework\v4.0.30319\aspnet_regiis.exe -i 一.找到C:\WINNT\MICROSOFT.NET\FRAMEWORK\V版本号\ASPNET_REGIIS.EXE 直接双击运行 即可,好像通过DOS命令不行. 二.开始- 程