1. 引入web
同步,任务以固定的顺序执行。异步,任务不须要按照固定顺序执行。从.net4.5开始,异步主要经过Task实现。异步编程的核心是Task和Task<T>对象,用来模拟异步操做,并经过async和await关键字修饰。其中异步模型的操做,通常分为如下2种:数据库
(1)对于I/O绑定的代码,await将在异步方法中返回一个Task或Task<T>;编程
(2)对于绑定CPU类的程序,await将在带有Task.Run()方法的后台线程上启动操做;缓存
2. 异步编程的三种模式异步
这三种模式的实现,以"从指定的偏移量开始将指定数量的数据读入提供的缓存区"为例进行实现。async
public class MyClass { public int Read(byte[] buffer, int offset, int count); }
该方法的TAP模式实现以下:异步编程
public class MyClass { public Task<int> ReadAsync(byte[] buffer, int offset, int count); }
EAP对应方法将公开一下类型和成员:测试
public class MyClass { public void ReadAsync(byte [] buffer, int offset, int count); public event ReadCompletedEventHandler ReadCompleted; }
APM模式下实现以下:spa
public class MyClass { public IAsyncResult BeginRead(byte [] buffer,int offset, int count,AsyncCallback callback, object state); public int EndRead(IAsyncResult asyncResult); }
3. 异步操做的实现.net
3.1 I/O绑定实例
场景描述: 点击按钮从web服务下载数据,但不能锁定UI线程。
private readonly HttpClient _httpClient=new HttpClient(); downloadButton.Click +=async (o,e)=> { //在web服务请求发生时间向UI提供控制 //UI线程自由地执行其余工做 var stringData = await _httpClient.GetStringAsync(URL); DoSomethingWithData(stringData); }
3.2 CPU绑定实例:执行游戏计算
场景描述:写一个手机端游戏,点击按钮对屏幕上许多敌人形成伤害,但执行损害计算的代价多是昂贵的,而且在UI线程上执行计算时可能致使界面暂停。
解决该问题的最佳办法是开启一个用Task.Run工做的后台线程,用await等待返回结果。
private DamageResult CalculateDamageDone() { // Code omitted: // // Does an expensive calculation and returns // the result of that calculation. } calculateButton.Clicked += async (o, e) => { // This line will yield control to the UI while CalculateDamageDone() // performs its work. The UI thread is free to perform other work. var damageResult = await Task.Run(() => CalculateDamageDone()); DisplayDamage(damageResult); };
3.3 关键点
3.4 如何分辨是I/O绑定仍是CPU绑定?
(1)若是代码须要“等待”,例如来自数据库的数据,则为I/O绑定;
(2)若是代码执行须要消耗大代价的计算,则为CPU绑定。
4. 总结