[C#] 走进异步编程的世界 - 剖析异步方法(下)
走进异步编程的世界 - 剖析异步方法(下)
序
感谢大家的支持,这是昨天发布《走进异步编程的世界 - 剖析异步方法(上)》的补充篇。
目录
一、异常处理
await 表达式也可以使用 try...catch...finally 结构。
internal class Program
{
private static void Main(string[] args)
{
var t = DoExceptionAsync();
t.Wait(); Console.WriteLine($"{nameof(t.Status)}: {t.Status}"); //任务状态
Console.WriteLine($"{nameof(t.IsCompleted)}: {t.IsCompleted}"); //任务完成状态标识
Console.WriteLine($"{nameof(t.IsFaulted)}: {t.IsFaulted}"); //任务是否有未处理的异常标识 Console.Read();
} /// <summary>
/// 异常操作
/// </summary>
/// <returns></returns>
private static async Task DoExceptionAsync()
{
try
{
await Task.Run(() => { throw new Exception(); });
}
catch (Exception)
{
Console.WriteLine($"{nameof(DoExceptionAsync)} 出现异常!");
}
}
}

图1-1
【分析】await 表达式位于 try 块中,按普通的方式处理异常。但是,为什么图中的状态(Status)、是否完成标识(IsCompleted)和是否失败标识(IsFaulted)分别显示:运行完成(RanToCompletion) 、已完成(True) 和 未失败(False) 呢?因为:任务没有被取消,并且异常都已经处理完成!
二、在调用方法中同步等待任务
调用方法可能在某个时间点上需要等待某个特殊的 Task 对象完成,才执行后面的代码。此时,可以采用实例方法 Wait 。
internal class Program
{
private static void Main(string[] args)
{
var t = CountCharactersAsync("http://www.cnblogs.com/liqingwen/"); t.Wait(); //等待任务结束
Console.WriteLine($"Result is {t.Result}"); Console.Read();
} /// <summary>
/// 统计字符数量
/// </summary>
/// <param name="address"></param>
/// <returns></returns>
private static async Task<int> CountCharactersAsync(string address)
{
var result = await Task.Run(() => new WebClient().DownloadStringTaskAsync(address));
return result.Length;
}
}

图2-1
Wait() 适合用于单一 Task 对象,如果想操作一组对象,可采用 Task 的两个静态方法 WaitAll() 和 WaitAny() 。
internal class Program
{
private static int time = ;
private static void Main(string[] args)
{
var t1 = GetRandomAsync();
var t2 = GetRandomAsync(); //IsCompleted 任务完成标识
Console.WriteLine($"t1.{nameof(t1.IsCompleted)}: {t1.IsCompleted}");
Console.WriteLine($"t2.{nameof(t2.IsCompleted)}: {t2.IsCompleted}"); Console.Read();
} /// <summary>
/// 获取一个随机数
/// </summary>
/// <param name="id"></param>
/// <returns></returns>
private static async Task<int> GetRandomAsync(int id)
{
var num = await Task.Run(() =>
{
time++;
Thread.Sleep(time * );
return new Random().Next();
}); Console.WriteLine($"{id} 已经调用完成");
return num;
}
}

图2-2 两个任务的 IsCompleted 属性都显示未完成
现在,在 Main() 方法中新增两行代码(6 和 7 两行),尝试调用 WaitAll() 方法。
private static void Main(string[] args)
{
var t1 = GetRandomAsync();
var t2 = GetRandomAsync(); Task<int>[] tasks = new Task<int>[] { t1, t2 };
Task.WaitAll(tasks); //等待任务全部完成,才继续执行 //IsCompleted 任务完成标识
Console.WriteLine($"t1.{nameof(t1.IsCompleted)}: {t1.IsCompleted}");
Console.WriteLine($"t2.{nameof(t2.IsCompleted)}: {t2.IsCompleted}"); Console.Read();
}

图2-3 两个任务的 IsCompleted 属性都显示 True
现在,再次将第 7 行改动一下,调用 WaitAny() 方法试试。
private static void Main(string[] args)
{
var t1 = GetRandomAsync();
var t2 = GetRandomAsync(); Task<int>[] tasks = new Task<int>[] { t1, t2 };
Task.WaitAny(tasks); //等待任一 Task 完成,才继续执行 //IsCompleted 任务完成标识
Console.WriteLine($"t1.{nameof(t1.IsCompleted)}: {t1.IsCompleted}");
Console.WriteLine($"t2.{nameof(t2.IsCompleted)}: {t2.IsCompleted}"); Console.Read();
}

图2-4 有一个任务的 IsCompleted 属性显示 True (完成) 就继续执行
三、在异步方法中异步等待任务
上节说的是如何使用 WaitAll() 和 WaitAny() 同步地等待 Task 完成。这次我们使用 Task.WhenAll() 和 Task.WhenAny() 在异步方法中异步等待任务。
internal class Program
{
private static int time = ; private static void Main(string[] args)
{
var t = GetRandomAsync(); Console.WriteLine($"t.{nameof(t.IsCompleted)}: {t.IsCompleted}");
Console.WriteLine($"Result: {t.Result}"); Console.Read();
} /// <summary>
/// 获取一个随机数
/// </summary>
/// <param name="id"></param>
/// <returns></returns>
private static async Task<int> GetRandomAsync()
{
time++;
var t1 = Task.Run(() =>
{
Thread.Sleep(time * );
return new Random().Next();
}); time++;
var t2 = Task.Run(() =>
{
Thread.Sleep(time * );
return new Random().Next();
}); //异步等待集合内的 Task 都完成,才进行下一步操作
await Task.WhenAll(new List<Task<int>>() { t1, t2 }); Console.WriteLine($" t1.{nameof(t1.IsCompleted)}: {t1.IsCompleted}");
Console.WriteLine($" t2.{nameof(t2.IsCompleted)}: {t2.IsCompleted}"); return t1.Result + t2.Result;
}
}

图3-1 调用 WhenAll() 方法
【注意】WhenAll() 异步等待集合内的 Task 都完成,不会占用主线程的时间。
现在,我们把 GetRandomAsync() 方法内的 WhenAll() 方法替换成 WhenAny(),并且增大一下线程挂起时间,最终改动如下:
private static async Task<int> GetRandomAsync()
{
time++;
var t1 = Task.Run(() =>
{
Thread.Sleep(time * );
return new Random().Next();
}); time++;
var t2 = Task.Run(() =>
{
Thread.Sleep(time * ); //这里由 100 改为 500,不然看不到效果
return new Random().Next();
}); //异步等待集合内的 Task 都完成,才进行下一步操作
//await Task.WhenAll(new List<Task<int>>() { t1, t2 });
await Task.WhenAny(new List<Task<int>>() { t1, t2 }); Console.WriteLine($" t1.{nameof(t1.IsCompleted)}: {t1.IsCompleted}");
Console.WriteLine($" t2.{nameof(t2.IsCompleted)}: {t2.IsCompleted}"); return t1.Result + t2.Result;
}

图3-2 调用 WhenAny() 方法
四、Task.Delay() 暂停执行
Task.Delay() 方法会创建一个 Task 对象,该对象将暂停其在线程中的处理,并在一定时间之后完成。和 Thread.Sleep 不同的是,它不会阻塞线程,意味着线程可以继续处理其它工作。
internal class Program
{
private static void Main(string[] args)
{
Console.WriteLine($"{nameof(Main)} - start.");
DoAsync();
Console.WriteLine($"{nameof(Main)} - end."); Console.Read();
} private static async void DoAsync()
{
Console.WriteLine($" {nameof(DoAsync)} - start."); await Task.Delay(); Console.WriteLine($" {nameof(DoAsync)} - end.");
}
}

图4-1
传送门
下篇:《走进异步编程的世界 - 在 GUI 中执行异步操作》
【原文链接】http://www.cnblogs.com/liqingwen/p/5866241.html
【参考】《Illustrated C# 2012》
[C#] 走进异步编程的世界 - 剖析异步方法(下)的更多相关文章
- [C#] 走进异步编程的世界 - 剖析异步方法(上)
走进异步编程的世界 - 剖析异步方法(上) 序 这是上篇<走进异步编程的世界 - 开始接触 async/await 异步编程>(入门)的第二章内容,主要是与大家共同深入探讨下异步方法. 本 ...
- [C#] 走进异步编程的世界 - 开始接触 async/await
走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...
- [C#] 走进异步编程的世界 - 在 GUI 中执行异步操作
走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5877042.html 序 这是继<开始接 ...
- 走进异步编程的世界 - 开始接触 async/await(转)
序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $"" 来拼接字符串,相当于string.Fo ...
- 走进异步编程的世界 - 在 GUI 中执行异步操作
转载:https://www.cnblogs.com/liqingwen/p/5877042.html 走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://w ...
- [C#] 走进异步编程的世界 - 开始接触 async/await(转)
原文链接:http://www.cnblogs.com/liqingwen/p/5831951.html 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 ...
- 走进异步编程的世界 - 开始接触 async/await
[C#] 走进异步编程的世界 - 开始接触 async/await 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async ...
- 走进异步编程的世界--async/await项目使用实战
起因:今天要做一个定时器任务:五分钟查询一次数据库发现超时未支付的订单数据将其状态改为已经关闭(数据量大约100条的情况) 开始未使用异步: public void SelfCloseGpPayOrd ...
- javascript异步编程方案汇总剖析
code[class*="language-"] { padding: .1em; border-radius: .3em; white-space: normal; backgr ...
随机推荐
- Android请求网络共通类——Hi_博客 Android App 开发笔记
今天 ,来分享一下 ,一个博客App的开发过程,以前也没开发过这种类型App 的经验,求大神们轻点喷. 首先我们要创建一个Andriod 项目 因为要从网络请求数据所以我们先来一个请求网络的共通类. ...
- 理解Maven中的SNAPSHOT版本和正式版本
Maven中建立的依赖管理方式基本已成为Java语言依赖管理的事实标准,Maven的替代者Gradle也基本沿用了Maven的依赖管理机制.在Maven依赖管理中,唯一标识一个依赖项是由该依赖项的三个 ...
- 一步步开发自己的博客 .NET版(10、前端对话框和消息框的实现)
关于前端对话框.消息框的优秀插件多不胜数.造轮子是为了更好的使用轮子,并不是说自己造的轮子肯定好.所以,这个博客系统基本上都是自己实现的,包括日志记录.响应式布局.评论功能等等一些本可以使用插件的.好 ...
- C#4.0泛型的协变,逆变深入剖析
C#4.0中有一个新特性:协变与逆变.可能很多人在开发过程中不常用到,但是深入的了解他们,肯定是有好处的. 协变和逆变体现在泛型的接口和委托上面,也就是对泛型参数的声明,可以声明为协变,或者逆变.什么 ...
- SQL Server常见数据类型介绍
数据表是由多个列组成,创建表时必须明确每个列的数据类型,以下列举SQL Server常见数据类型的使用规则,方便查阅. 1.整数类型 int 存储范围是-2,147,483,648到2,147,483 ...
- Android中Activity的四大启动模式实验简述
作为Android四大组件之一,Activity可以说是最基本也是最常见的组件,它提供了一个显示界面,从而实现与用户的交互,作为初学者,必须熟练掌握.今天我们就来通过实验演示,来帮助大家理解Activ ...
- LINQ to SQL语句(7)之Exists/In/Any/All/Contains
适用场景:用于判断集合中元素,进一步缩小范围. Any 说明:用于判断集合中是否有元素满足某一条件:不延迟.(若条件为空,则集合只要不为空就返回True,否则为False).有2种形式,分别为简单形式 ...
- Spring MVC注解开发入门
注解式开发初步 常用的两个注解: @Controller:是SpringMVC中最常用的注解,它可以帮助定义当前类为一个Spring管理的bean,同时指定该类是一个控制器,可以用来接受请求.标识当前 ...
- Spring Security OAuth2 开发指南
官方原文:http://projects.spring.io/spring-security-oauth/docs/oauth2.html 翻译及修改补充:Alex Liao. 转载请注明来源:htt ...
- 【Knockout.js 学习体验之旅】(3)模板绑定
本文是[Knockout.js 学习体验之旅]系列文章的第3篇,所有demo均基于目前knockout.js的最新版本(3.4.0).小茄才识有限,文中若有不当之处,还望大家指出. 目录: [Knoc ...