谈到C#中的异步编程,离不开Async和Await关键字编程
谈到异步编程,首先咱们就要明白到底什么是异步编程。 平时咱们的编程通常都是同步编程,所谓同步编程的意思,和咱们平时说的同时作几件事情彻底不一样。 在计算机的世界里,同步编程的意思说 按照顺序来执行,或者说是 一个接着一个地有序的来执行, 好比目前咱们在代码中有三件任务来执行,那么必须先执行完第1件,再执行第2件,接下来再执行第3件。 在这个过程当中,第1件没有完成,你是无法开始作第2件事情的,必须等待。异步
好比一我的烧开水须要10分钟,5分钟找杯子,5分钟找茶叶。 这件事情在同步编程的世界里须要20分钟完成,由于必须先烧开水10分钟,等烧完开水后。 才开始作第2件事情找杯子,花费5分钟。 找完杯子后,才开始作第3件事情-找茶叶,又花费5分钟。这样,整个过程就花费了20分钟。但其实这3件事情能够同时作,互相并不影响,这就是异步编程的概念了async
由上面这个例子中能够发现,在计算机编程世界里,【异步编程】才是真正的同时进行事情,而【同步编程】是一件一件的有序执行。 上面的例子,若是采用异步编程,那么10分钟就能够所有完成。异步编程
在C#中,异步编程中有三个方面须要注意:this
1.为了表示一个方法是异步方法,须要使用async关键字来修饰该方法签名spa
2. 异步方法的返回值类型 --- 只有 void, Task 和 Task<T>线程
3. 在异步方法内部,你须要使用await关键字来修饰一个能够等待的【可等待】类型,来实现异步code
咱们经过例子来比较同步编程和异步编程的不一样blog
举例: 好比页面上有个【计算】按钮,点击该按钮 程序将会进行一个复杂运算,该运算将花费45秒的时间,运算完后返回计算结果,现实在页面的一个textbox文本框里面同步
private void btnCalculate_Click(object sender, EventArgs e) { int calResult = ToCalculate(); this. Textbox1.Text = calResult; } private int ToCalculate() { // 这里有复杂的计算过程,将耗时45秒, 咱们这里经过让线程休眠45秒钟来模拟 System.Threading.Thread.Sleep(45000); return 100; }
在上面代码中,咱们假设方法ToCalculate方法要进行复杂的计算。在这里,咱们经过让当前线程挂起45秒,来模拟耗时45秒的复杂计算过程.
咱们能够想象获得,上面得代码中,当用户在界面上点击btnCalculate按钮时,UI在接下来的45秒里将毫无反应,程序将只干一件事情,就是容许ToCalculate方法来进行耗时45秒的复杂计算。45秒以后,才会把计算结果返回到Textbox1中,再接下来作其余的操做
这个显然很是的用户不友好,由于用户会发现他须要等在那里45秒,什么事情也不能干。若是在这个等待过程当中,用户在UI上还能同时操做干其余的事情呢? 这个就须要涉及到异步编程的概念
【异步编程】的操做过程以下:
执行异步方法,当遇到await关键字时,表示要开始执行一个异步方法,await关键字来修饰的方法是一个方法。这个时候程序会作两件事情
第1件: 程序去执行await关键字修饰的异步方法
第2件: 在作第1件的同时,把控制权当即返回给调用者,也就是说调用者这个时候,能够在等待异步方法执行的过程当中,同时去作其余事情,由于控制权已经在它手中了。
看看上面的例子,若是采用异步方法,当用户在界面上点击btnCalculate按钮时,程序会执行异步方法ToCalculate. 但与此同时,控制权也立刻回到UI手上,也就是说在后台进行耗时45秒的复杂计算的同时,用户能够同时在UI上进行其余操做。当异步计算结束,把计算结果显示在UI的Textbox1上。
了解了整个过程后,咱们使用异步方法来改写上面的代码
private async void btnCalculate_Click(object sender, EventArgs e) { int calResult = await ToCalculate(); this. Textbox1.Text = calResult; } private Task<int> ToCalculate() { //将复杂计算的过程放入一个Task<int>中,新开线程 var t = Task.Run(()=> { System.Threading.Thread.Sleep(45000); return 100; }); //返回的是这个Task<int> return t; }
从上面的代码变更中,咱们能够看到几处变更状况
1. 在方法的签名中,在public后面加入了async关键字, 将该方法标识成一个异步方法。 这样,让程序知道这个方法是一个异步方法, 此时它内部的await才会被认为是一个关键字。
或者说,若是在方法的签名中没有async关键字, await会被认为是一个普通的变量名,也就是说,它会被当成普通的C#标识符来处理(相似于 string await ="test";)
2. 在async标识的异步方法中,在须要异步调用的方法ToCalculate方法前面,使用await关键字
3. 异步调用的方法ToCalculate,只能是三种返回值类型中的一种void, Task 和 Task<T>。 在这个例子中,是Task<T>