隐藏

C# 线程知识--使用Task执行异步操作

发布:2019/12/17 22:27:46作者:管理员 来源:本站 浏览次数:1067

原文地址:http://www.cnblogs.com/pengstone/archive/2012/12/23/2830238.html

感觉这篇文章不错,copy了一份;

   在C#4.0之前需要执行一个复杂的异步操作时,只能使用CLR线程池技术来执行一个任务。线程池执行异步任务时,不知道任务何时完成,以及任务的在任务完成后不能获取到返回值。但是在C#4.0中引人了一个的任务(System.Threading.Tasks命名空间的类型)机制来解决异步操作完成时间和完成后返回值的问题。

1.使用Task类创建并执行简单任务

通过使用Task的构造函数来创建任务,并调用Start方法来启动任务并执行异步操作。创建任务时,必须传递一个Action或Action<Object>类型的委托回调方法,可以选择的传递任务执行时说需要的数据对象等。Task类的构造函数如下:

 


			
  1. public Task(Action action);
  2. public Task(Action<object> action, object state);
  3. public Task(Action action, CancellationToken cancellationToken);
  4. public Task(Action action, TaskCreationOptions creationOptions);
  5. public Task(Action<object> action, object state, CancellationToken cancellationToken);
  6. public Task(Action<object> action, object state, TaskCreationOptions creationOptions);
  7. public Task(Action action, CancellationToken cancellationToken, TaskCreationOptions creationOptions);
  8. public Task(Action<object> action, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions);

 

示例代码:

 

			
  1. static void Main(string[] args)
  2. {
  3. //先声明一个委托
  4. Console.WriteLine("主线程执行业务处理---开始.");
  5. Task task = new Task(() =>
  6. {
  7. Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作----开始.");
  8. for (int i = 0; i < 10; i++)
  9. {
  10. Console.WriteLine(i);
  11. }
  12. Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作----结束.");
  13. });
  14. task.Start();
  15. Console.WriteLine("主线程执行其他处理---结束");
  16. //
  17. Console.ReadKey(false);
  18. }



结果:

主线程执行业务处理---开始.
主线程执行其他处理---结束
使用System.Threading.Tasks.Task执行异步操作----开始.
0
1
2
3
4
5
6
7
8
9
使用System.Threading.Tasks.Task执行异步操作----结束.

 2.等待任务的完成并获取返回值

使用任务执行异步操作时,最主要的是要后的任务完成时的返回值。在任务类中有一个实例方法Wait(有许多重载版本)他能等待任务的完成,我们也可以通过Task类的派生类Task<TResult>创建一个异步任务,并指定任务完成时返回值的类型,这样可以通过Task<TResult>的实例对象获取到任务完成后的返回值。创建一个异步任务并执行0到100求和操作返回最后的计算结果,示例代码:


			
  1. Console.WriteLine("主线程执行业务处理---开始.");
  2. Task<int> task1 = new Task<int>(()=> {
  3. Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作----开始.");
  4. int sum = 0;
  5. for (int i = 0; i < 50; i++)
  6. {
  7. sum += i;
  8. }
  9. return sum;
  10. });
  11. task1.Start();
  12. Console.WriteLine("主线程执行其他处理---结束");
  13. //等待任务的完成执行过程。
  14. task1.Wait();
  15. Console.WriteLine("任务执行结果:{0}", task1.Result.ToString());

执行结果:

主线程执行业务处理---开始.
主线程执行其他处理---结束
使用System.Threading.Tasks.Task执行异步操作----开始.
任务执行结果:1225

 Task类还有一些静态方法,WaitAll用于等待提供的所有 System.Threading.Tasks.Task 对象完成执行过程和Wait用于等待提供的任一个 System.Threading.Tasks.Task 对象完成执行过程,这两个方法都有一些重载版本。


			
  1. //等待所有任务完成
  2. public static void WaitAll(params Task[] tasks);
  3. //等待任意一个任务完成
  4. public static int WaitAny(params Task[] tasks);

3.使用ContinueWith方法在任务完成时启动一个新任务

在使用能够Task类的Wait方法等待一个任务时或派生类的Result属性获得任务执行结果都有可能阻塞线程,为了解决这个问题可以使用ContinueWith方法,他能在一个任务完成时自动启动一个新的任务来处理执行结果。

示例代码:

 


			
  1. Console.WriteLine("主线程执行业务处理---开始.");
  2. Task<int> task2 = new Task<int>(() => {
  3. Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作----开始.");
  4. int sum = 0;
  5. for (int i = 0; i < 50; i++)
  6. {
  7. sum += i;
  8. }
  9. return sum;
  10. });
  11. task2.Start();
  12. Console.WriteLine("主线程执行其他处理---结束");
  13. task2.ContinueWith(r=>{
  14. Console.WriteLine("任务执行结果:{0}", r.Result.ToString());
  15. });

 


 

执行结果:

主线程执行业务处理---开始.
主线程执行其他处理---结束
使用System.Threading.Tasks.Task执行异步操作----开始.
任务执行结果:1225

 4.创建父子任务和任务工厂的使用

通过Task类创建的任务是顶级任务,可以通过使用 TaskCreationOptions.AttachedToParent 标识把这些任务与创建他的任务相关联,所有子任务全部完成以后父任务才会结束操作。示例如下:


			
  1. public static void ParentChildTask()
  2. {
  3. Task<string[]> parent = new Task<string[]>(state =>
  4. {
  5. Console.WriteLine(state);
  6. string[] result = new string[2];
  7. //创建并启动子任务
  8. new Task(() => { result[0] = "我是子任务1。"; }, TaskCreationOptions.AttachedToParent).Start();
  9. new Task(() => { result[1] = "我是子任务2。"; }, TaskCreationOptions.AttachedToParent).Start();
  10. return result;
  11. }, "我是父任务,并在我的处理过程中创建多个子任务,所有子任务完成以后我才会结束执行");
  12. parent.ContinueWith(t =>
  13. {
  14. Array.ForEach(t.Result, r => Console.WriteLine(r));
  15. });
  16. //启动父任务
  17. parent.Start();
  18. }

结果:

我是父任务,并在我的处理过程中创建多个子任务,所有子任务完成以后我才会结束执行
我是子任务1。
我是子任务2。

 

转载于:https://www.cnblogs.com/alphafly/p/5147843.html