一切福田,不離方寸,從心而覓,感無不通。

说说C#的async和await

C# 5.0中引入了async 和 await。这两个关键字可以让你更方便的写出异步代码。

看个例子:

  1. public class MyClass
  2. {
  3.     public MyClass()
  4.     {
  5.         DisplayValue(); //这里不会阻塞
  6.         System.Diagnostics.Debug.WriteLine("MyClass() End.");
  7.     }
  8.     public Task<double> GetValueAsync(double num1, double num2)
  9.     {
  10.         return Task.Run(() =>
  11.         {
  12.             for (int i = 0; i < 1000000; i++)
  13.             {
  14.                 num1 = num1 / num2;
  15.             }
  16.             return num1;
  17.         });
  18.     }
  19.     public async void DisplayValue()
  20.     {
  21.         double result = await GetValueAsync(1234.5, 1.01);//此处会开新线程处理GetValueAsync任务,然后方法马上返回
  22.         //这之后的所有代码都会被封装成委托,在GetValueAsync任务完成时调用
  23.         System.Diagnostics.Debug.WriteLine("Value is : " + result);
  24.     }
  25. }

上面在MyClass的构造函数里调用了async关键字标记的异步方法DisplayValue(),DisplayValue()方法里执行了一个await关键字标记的异步任务GetValueAsync(),这个异步任务必须是以Task或者Task<TResult>作为返回值的,而我们也看到,异步任务执行完成时实际返回的类型是void或者TResult,DisplayValue()方法里await GetValueAsync()之后的所有代码都会在异步任务完成时才会执行。

DisplayValue()方法实际执行的代码如下:

  1. public void DisplayValue()
  2. {
  3.     System.Runtime.CompilerServices.TaskAwaiter<double> awaiter = GetValueAsync(1234.5, 1.01).GetAwaiter();
  4.     awaiter.OnCompleted(() =>
  5.         {
  6.             double result = awaiter.GetResult();
  7.             System.Diagnostics.Debug.WriteLine("Value is : " + result);
  8.         });
  9. }

可以看到,async和await关键字只是把上面的代码变得更简单易懂而已。

程序的输出如下:

MyClass() End.

Value is : 2.47032822920623E-322

以下是我写的一个静态类,可以方便将一个普通Function执行异步调用:

  1. public static class TaskAsyncHelper
  2. {
  3.     /// <summary>
  4.     /// 将一个方法function异步运行,在执行完毕时执行回调callback
  5.     /// </summary>
  6.     /// <param name="function">异步方法,该方法没有参数,返回类型必须是void</param>
  7.     /// <param name="callback">异步方法执行完毕时执行的回调方法,该方法没有参数,返回类型必须是void</param>
  8.     public static async void RunAsync(Action function, Action callback)
  9.     {
  10.         Func<System.Threading.Tasks.Task> taskFunc = () =>
  11.         {
  12.             return System.Threading.Tasks.Task.Run(() =>
  13.             {
  14.                 function();
  15.             });
  16.         };
  17.         await taskFunc();
  18.         if (callback != null)
  19.             callback();
  20.     }
  21.     /// <summary>
  22.     /// 将一个方法function异步运行,在执行完毕时执行回调callback
  23.     /// </summary>
  24.     /// <typeparam name="TResult">异步方法的返回类型</typeparam>
  25.     /// <param name="function">异步方法,该方法没有参数,返回类型必须是TResult</param>
  26.     /// <param name="callback">异步方法执行完毕时执行的回调方法,该方法参数为TResult,返回类型必须是void</param>
  27.     public static async void RunAsync<TResult>(Func<TResult> function, Action<TResult> callback)
  28.     {
  29.         Func<System.Threading.Tasks.Task<TResult>> taskFunc = ()=>
  30.             {
  31.                 return System.Threading.Tasks.Task.Run(()=>
  32.                     {
  33.                         return function();
  34.                     });
  35.             };
  36.         TResult rlt = await taskFunc();
  37.         if(callback != null)
  38.             callback(rlt);
  39.     }
  40. }

使用很简单,将方法名作为参数传进去就行了,最常用的是把很耗时的序列化函数传进去,以免阻塞UI进程,造成卡顿现象,影响用户体验。

from:http://blog.csdn.net/tianmuxia/article/details/17675681/