1 异步编程的重要性

  C#5.0最重要的改进是提供了更强大的异步编程,C#5.0仅增加两个关键字Async和Await,使用异步编程,方法调用是后台运行(通常在线程和任务的帮助下),并且不会阻塞调用线程。

2 异步模式

  从.net1.0开始就提供了异步特性,而且.NET Framework的许多类都实现了一个或多个异步模式(委托也实现了异步模式)。因为在WIndows From和WPF中用异步模式更新界面比较复杂,所以在2.0中提供了基于事件的异步模式。现在在4.5中推出了另外一种新的方式来实现异步编程:基于任务的异步模式。这种模式是基于4.0中新增的Task类型,和一个利用Async和Await关键字的编译器功能。

2.1 同步调用

WebClient wc = string s= wc.DownloadString("你的URL");

  运行该代码,程序一直在等待DownloadString的完成,在这种情况下使用异步非常有必要的。

2.2异步模式

使用异步模式是进行异步调用的方式之一,实现异步模式定义BeginXXX和EndXXX方法。如果有一个同步方法DownloadStrring,异步方法将转化成了两个方法BeginDownloadString和EndDownloadString方法。BeginXXX方法接受同步方法的所有输入的参数,并且还定义了一个AsyncCallBack参数,用于接收在异步方法执行完毕后的被调用的委托,该方法返回一个IAsyncResult,用于验证调用是否已经完成,并且一直等待,直到方法的执行完毕。EndXXX方法同步方法的所有的输出参数,并按照同步方法的返回类型来返回结果。WebClient类没有实现异步模式,但可以用HttpWebRequest来代替

  HttpWebRequest req =(HttpWebRequest)WebRequest.Create("你的URL");
             IAsyncResult result=  req.BeginGetResponse("",);
             req.EndGetResponse();

异步模式的优势是使用委托功能就能实现异步编程,不用改变程序的行为,也不会阻塞界面的操作。

2.3基于事件的异步模式

  基于事件的异步模式定义了一个带有"Aysnc"后缀的方法。异步方法完成时不是定义被调用的委托,而是定义事件。

 WebClient client = new WebClient();
             client.DownloadStringCompleted += (sender, e1) => { };
             client.DownloadStringAsync(new Uri("你的URL"));

基于事件的异步方式的优势是易用。

2.4基于任务的异步模式

 public async void Fun()
         {
             WebClient client = new WebClient();
             string s = await client.DownloadStringTaskAsync("你的URL");
         }

现在代码简单多了,并且没有阻塞。

3 异步编程的基础

  async和await关键字只是编译器功能,编译器会用Task创建代码。

3.1创建任务

         static string Greeting(string name)
         {
             Thread.Sleep();
             return string.Format("Hello ,{0}", name);
         }
         static Task<string> GreetingAsync()
         {
             return Task.Run<string>(()=> { return Greeting(name); });
         }

3.2调用异步方法

  可以使用Await关键字来调用返回任务的异步方法GreetingAsync,使用Await关键字需要使用Async修饰声明的方法。在完成GreetingAsync方法前该方法内的其它代码不会被执行。

   private async static void CallWithAsync()
         {
             string result = await GreetingAsync("郑隆");
             Console.WriteLine(result);
         }

如果异步方法的结果不传递给变量,也可以直接在参数中使用Await关键字。

  private async static void CallWithAsync2()
         {
             Console.WriteLine(await GreetingAsync("郑隆"));
         }
Async修饰符只能用于返回Task和void的方法。不能用于程序的入口。

3.3延续任务

Task类的ContinueWith方法定义了任务完成后将要调用的代码。

    private static void CallerWithContinuationTask()
         {
             Task<string> t1 = GreetingAsync("郑隆");
             t1.ContinueWith(t => { string result = t.Result; Console.WriteLine(result); });
         }

3.4使用多个异步方法

在一个异步方法中,可以调用不止一个异步方法。

  public async static void MultipleAsyncMethods()
         {
             string s1 = await GreetingAsync("zhenglong1");
             string s2 = await GreetingAsync("zhenglong2");
             Console.WriteLine("Finished both methods.\n"+"Result 1:{0}\n Result 2:{1}",s1,s2);
         }

Task.WhenAll组合器,可以让你等待直到两个任务都完成。

  public async static void MultipleAsyncMethodsWithCombinators1()
         {
             Task<string>  t1= GreetingAsync("zhenglong1");
             Task<string> t2 = GreetingAsync("zhenglong2");
             await Task.WhenAll(t1,t2);
             Console.WriteLine("Finished both methods.\n" + "Result 1:{0}\n Result 2:{1}", t1.Result, t2.Result);
         }

Task的WhenAll方法是在所有传入的任务都完成了才返回Task,而WhenAny是在其中的一个任务已完成就会返回Task。

3.5转换异步模式

并非.NET Freamwork的所有的类在4.5中都引入了新的异步方法。

 private static Func<string, string> greetingInvoker = Greeting;

         static IAsyncResult BeginGreeting(string name,AsyncCallback callback,object state)
         {
             return greetingInvoker.BeginInvoke(name, callback, state);
         }
         static string EndGreeting(IAsyncResult ar)
         {
            return greetingInvoker.EndInvoke(ar);
         }
  private async static void ConvertingAsyncPattern()
         {
             string s = await Task<string>.Factory.FromAsync(BeginGreeting, EndGreeting, "zhenglong", null);
             Console.WriteLine(s);
         }

4 错误处理

   public static async Task ThrowAfter(int ms,string message)
         {
             await Task.Delay(ms);
             throw new Exception(message);
         }

         public static void DontHandle()
         {
             try
             {
                 ThrowAfter(,"first");
             }
             catch (Exception ex)
             {
              }
         }

以上代码并不能捕获到异常,应为在ThrowAfter抛出异常之前DontHandle已经执行完了。

4.1 异步方法的异常处理

异步方法异常的一个较好的处理是使用关键字Await,然后将其放在try中

    public async static void HandleErrorOne()
         {
             try
             {
                 , "first");
             }
             catch (Exception ex)
             {
                 Console.WriteLine(ex.Message);
             }
         }

4.2 多个异步方法的异常处理

   public static async void StatTwoTasks()
         {
             try
             {
                 ,"frist");
                 , "second");
             }
             catch (Exception ex)
             {
                 Console.WriteLine(ex.Message);
             }
         }

如上代码并不能捕获全部的异常,原因是因为在第一个异常抛出后程序就进入了catch。解决方法

  public static async void StatTwoTasksParallel()
         {
             try
             {
                 Task t1 = ThrowAfter(, "zhenglong1");
                 Task t2 = ThrowAfter(, "zhenglong2");
                 await Task.WhenAll(t1,t2);
             }
             catch (Exception ex)
             {
                 Console.WriteLine(ex.Message);
             }
         }

4.3 AggregateException类

为了得到所有的异常信息,可以将Task.WhenAll返回的结果写到一个Task中,这个任务一直等待所有的任务完成。

 public static async void ShowAggregateException()
         {
             Task taskResult = null;
             try
             {
                 Task t1 = ThrowAfter(, "zhenglong1");
                 Task t2 = ThrowAfter(, "zhenglong2");
                 await(taskResult= Task.WhenAll(t1, t2));
             }
             catch (Exception ex)
             {
                 Console.WriteLine(ex.Message);
                 foreach (var item in taskResult.Exception.InnerExceptions)
                 {
                     Console.WriteLine(item.Message);
                 }
             }
         }

5取消

在某种情况下,后台任务可能运行很长的时间,取消任务就非常有用了。

5.1 开始取消任务

 private CancellationTokenSource cts;
         public void Cancle()
         {
             if (cts != null)
                 cts.Cancel();
         }
CancellationTokenSource 类还支持在指定时间后才取消任务,CancelAfter方法在应该取消的任务后传入一个时间值,单位是ms.

5.2开始框架的特性任务取消

  public async void CancelTwo()
         {
             cts = new CancellationTokenSource();
             HttpClient hc = new HttpClient();
             var s=await hc.GetAsync("Url", cts.Token);
         }

5.3取消自定义任务

  await Task.Run(()=> { },cts.Token);

现在用户就可以取消运行时间长的任务了。

C#之异步编程的更多相关文章

  1. C#异步编程(一)

    异步编程简介 前言 本人学习.Net两年有余,是第一次写博客,虽然写的很认真,当毕竟是第一次,肯定会有很多不足之处, 希望大家照顾照顾新人,有错误之处可以指出来,我会虚心接受的. 何谓异步 与同步相对 ...

  2. C#与C++的发展历程第三 - C#5.0异步编程巅峰

    系列文章目录 1. C#与C++的发展历程第一 - 由C#3.0起 2. C#与C++的发展历程第二 - C#4.0再接再厉 3. C#与C++的发展历程第三 - C#5.0异步编程的巅峰 C#5.0 ...

  3. 关于如何提高Web服务端并发效率的异步编程技术

    最近我研究技术的一个重点是java的多线程开发,在我早期学习java的时候,很多书上把java的多线程开发标榜为简单易用,这个简单易用是以C语言作为参照的,不过我也没有使用过C语言开发过多线程,我只知 ...

  4. 异步编程 In .NET

    概述 在之前写的一篇关于async和await的前世今生的文章之后,大家似乎在async和await提高网站处理能力方面还有一些疑问,博客园本身也做了不少的尝试.今天我们再来回答一下这个问题,同时我们 ...

  5. C#异步编程(二)

    async和await结构 序 前篇博客异步编程系列(一) 已经介绍了何谓异步编程,这篇主要介绍怎么实现异步编程,主要通过C#5.0引入的async/await来实现. BeginInvoke和End ...

  6. [.NET] 利用 async & await 的异步编程

    利用 async & await 的异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/5922573.html  目录 异步编程的简介 异 ...

  7. [.NET] 怎样使用 async & await 一步步将同步代码转换为异步编程

    怎样使用 async & await 一步步将同步代码转换为异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6079707.html  ...

  8. [C#] 走进异步编程的世界 - 开始接触 async/await

    走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...

  9. C#异步编程

    什么是异步编程 什么是异步编程呢?举个简单的例子: using System.Net.Http; using System.Threading.Tasks; using static System.C ...

  10. 深入解析js异步编程利器Generator

    我们在编写Nodejs程序时,经常会用到回调函数,在一个操作执行完成之后对返回的数据进行处理,我简单的理解它为异步编程. 如果操作很多,那么回调的嵌套就会必不可少,那么如果操作非常多,那么回调的嵌套就 ...

随机推荐

  1. C#分析URL参数获取参数和值得对应列表

    原文: C#分析URL参数获取参数和值得对应列表 /// <summary> /// 分析url链接,返回参数集合 /// </summary> /// <param n ...

  2. C++基础之二:常量指针和指针常量

    1.常量指针 定义:具有只能够读取内存中数据,却不能够修改内存中数据的属性的指针,称为指向常量的指针,简称常量指针. 声明:const int * p; 注:可以将一个常量的地址赋值给一个对应类型的常 ...

  3. cuzysdk购物模块 36kr+本期背景图

    说好的剧透,虽然来的稍微晚不少 cuzysdk(www.cuzy.com) 是一个手机淘宝客sdk,通过使用cuzy,可以获取taobao平台的推广商品数据,移动开发者把推广的商品数据呈现给用户,用户 ...

  4. keyset与entryset

    1.基本概述 Set<Map.Entry<K,V>> entrySet()  返回此映射中包括的映射关系的 set 视图. Set<K>              ...

  5. 轻量级工具网站SimpleTools

    [解释]本来这篇文章是在前天发出来的,可是当时是刚申请的域名,现在都要域名实名认证,导致我发的项目网址打不开,惹来了很多博友的吐槽,在此说声抱歉,今天一大早就把实名认证提交了,现在网站已经可以正常访问 ...

  6. javascript创建类的6种方式

    javascript创建类的7种方式 一 使用字面量创建 1.1 示例 var obj={}; 1.2 使用场景 比较适用于临时构建一个对象,且不关注该对象的类型,只用于临时封装一次数据,且不适合代码 ...

  7. win32 sdk树形控件的项拖拽实现

    本课中,我们将学习如何使用树型视图控件.另外还要学习如何在树型视图中完成拖-拉动作,以及如何使用图象列表. 理论: 树型视图是一种特别的窗口,我们可以使用它一目了然地表示某种层次关系.譬如象在资源管理 ...

  8. C++ 载入dll

    1.新建一个项目生成dll 首先我们新建一个项目生成一个Dynamic Library(动态链接库) dll 里面非常简单,只有一个add方法.等下我们就要在其他项目里尝试载入这个dll,调用里面的这 ...

  9. SRM 624 Building Heights DivI 解读

    几乎相同的一标题.欲了解更多请参阅:http://community.topcoder.com/stat?c=problem_statement&pm=13211&rd=15857 思 ...

  10. Maven-1:下载&安装

    (一)下载 下载网址:http://maven.apache.org/download.cgi 版本:3.0.5 (二)安装 1.解压至目录:F:\Kevin\software\apache-mave ...