走进异步编程的世界 - 剖析异步方法(上)

  这是上篇《走进异步编程的世界 - 开始接触 async/await 异步编程》(入门)的第二章内容,主要是与大家共同深入探讨下异步方法。

  本文要求了解委托的使用。

目录

介绍异步方法

     异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。
     语法分析:
     (1)关键字:方法头使用 async 修饰。
     (2)要求:包含 N(N>0) 个 await 表达式(不存在 await 表达式的话 IDE 会发出警告),表示需要异步执行的任务。【备注】感谢 czcz1024 的修正与补充:没有的话,就和普通方法一样执行了。
     (3)返回类型:只能返回 3 种类型(void、Task 和 Task<T>)。Task 和 Task<T> 标识返回的对象会在将来完成工作,表示调用方法和异步方法可以继续执行。
     (4)参数:数量不限。但不能使用 out 和 ref 关键字。
     (5)命名约定:方法后缀名应以 Async 结尾。
     (6)其它:匿名方法和 Lambda 表达式也可以作为异步对象;async 是一个上下文关键字;关键字 async 必须在返回类型前。
 
图1 异步方法的简单结构图

  关于 async 关键字:

  ①在返回类型之前包含 async 关键字

  ②它只是标识该方法包含一个或多个 await 表达式,即,它本身不创建异步操作。

  ③它是上下文关键字,即可作为变量名。

  现在先来简单分析一下这三种返回值类型:void、Task 和 Task<T>

  (1)Task<T>:调用方法要从调用中获取一个 T 类型的值,异步方法的返回类型就必须是Task<T>。调用方法从 Task 的 Result 属性获取的就是 T 类型的值。

         private static void Main(string[] args)
{
Task<int> t = Calculator.AddAsync(, ); //一直在干活 Console.WriteLine($"result: {t.Result}"); Console.Read();
}

Program.cs

     internal class Calculator
{
private static int Add(int n, int m)
{
return n + m;
} public static async Task<int> AddAsync(int n, int m)
{
int val = await Task.Run(() => Add(n, m)); return val;
}
}

图2

图3

  (2)Task:调用方法不需要从异步方法中取返回值,但是希望检查异步方法的状态,那么可以选择可以返回 Task 类型的对象。不过,就算异步方法中包含 return 语句,也不会返回任何东西。

         private static void Main(string[] args)
{
Task t = Calculator.AddAsync(, ); //一直在干活 t.Wait();
Console.WriteLine("AddAsync 方法执行完成"); Console.Read();
}

Program.cs

     internal class Calculator
{
private static int Add(int n, int m)
{
return n + m;
} public static async Task AddAsync(int n, int m)
{
int val = await Task.Run(() => Add(n, m));
Console.WriteLine($"Result: {val}");
}
}

图4

图5

  (3)void:调用方法执行异步方法,但又不需要做进一步的交互。

         private static void Main(string[] args)
{
Calculator.AddAsync(, ); //一直在干活 Thread.Sleep(); //挂起1秒钟
Console.WriteLine("AddAsync 方法执行完成"); Console.Read();
}

Program.cs

     internal class Calculator
{
private static int Add(int n, int m)
{
return n + m;
} public static async void AddAsync(int n, int m)
{
int val = await Task.Run(() => Add(n, m));
Console.WriteLine($"Result: {val}");
}
}

Calculator.cs

图6

图7

一、控制流

     异步方法的结构可拆分成三个不同的区域:
     (1)表达式之前的部分:从方法头到第一个 await 表达式之间的所有代码。
     (2)await 表达式:将被异步执行的代码。
     (3)表达式之后的部分:await 表达式的后续部分。
 
  图1-1
 
  该异步方法执行流程:从await表达式之前的地方开始,同步执行到第一个 await,标识着第一部分执行结束,一般来说此时 await 工作还没完成。当await 任务完成后,该方法将继续同步执行后续部分。在执行的后续部分中,如果依然存在 await,就重复上述过程。
  当到达 await 表达式时,线程将从异步方法返回到调用方法。如果异步方法的返回类型为 Task 或 Task<T>,会创建一个 Task 对象,标识需要异步完成的任务,然后将 Task 返回来调用方法。
 
  图1-2
  异步方法的控制流:
  ①异步执行 await 表达式的空闲任务。
  ②await 表达式执行完成,继续执行后续部分。如再遇到 await 表达式,按相同情况进行处理。
  ③到达末尾或遇到 return 语句时,根据返回类型可以分三种情况:
    a.void:退出控制流。
    b.Task:设置 Task 的属性并退出。
    c.Task<T>:设置 Task 的属性和返回值(Result 属性)并退出。
  ④同时,调用方法将继续执行,从异步方法获取 Task 对象。需要值的时候,会暂停等到 Task 对象的 Result 属性被赋值才会继续执行。
 
  【难点】
  ①第一次遇到 await 所返回对象的类型。这个返回类型就是同步方法头的返回类型,跟 await 表达式的返回值没有关系。
  ②到达异步方法的末尾或遇到 return 语句,它并没有真正的返回一个值,而是退出了该方法。
 

二、await 表达式

  await 表达式指定了一个异步执行的任务。默认情况,该任务在当前线程异步执行。

  每一个任务就是一个 awaitable 类的实例。awaitable 类型指包含 GetAwaiter() 方法的类型。

  实际上,你并不需要构建自己的 awaitable,一般只需要使用 Task 类,它就是 awaitable。

  最简单的方式是在方法中使用 Task.Run() 来创建一个 Task。【注意】它是在不同的线程上执行方法。

  让我们一起来看看示例。

     internal class Program
{
private static void Main(string[] args)
{
var t = Do.GetGuidAsync();
t.Wait(); Console.Read();
} private class Do
{
/// <summary>
/// 获取 Guid
/// </summary>
/// <returns></returns>
private static Guid GetGuid() //与Func<Guid> 兼容
{
return Guid.NewGuid();
} /// <summary>
/// 异步获取 Guid
/// </summary>
/// <returns></returns>
public static async Task GetGuidAsync()
{
var myFunc = new Func<Guid>(GetGuid);
var t1 = await Task.Run(myFunc); var t2 = await Task.Run(new Func<Guid>(GetGuid)); var t3 = await Task.Run(() => GetGuid()); var t4 = await Task.Run(() => Guid.NewGuid()); Console.WriteLine($"t1: {t1}");
Console.WriteLine($"t2: {t2}");
Console.WriteLine($"t3: {t3}");
Console.WriteLine($"t4: {t4}");
}
}
}

图2-1

图2-2

  上面 4 个 Task.Run() 都是采用了 Task Run(Func<TReturn> func) 形式来直接或间接调用 Guid.NewGuid()。

  Task.Run() 支持 4 中不同的委托类型所表示的方法:Action、Func<TResult>、Func<Task> 和 Func<Task<TResult>>

     internal class Program
{
private static void Main(string[] args)
{
var t = Do.GetGuidAsync();
t.Wait(); Console.Read();
} private class Do
{
public static async Task GetGuidAsync()
{
await Task.Run(() => { Console.WriteLine(Guid.NewGuid()); }); //Action Console.WriteLine(await Task.Run(() => Guid.NewGuid())); //Func<TResult> await Task.Run(() => Task.Run(() => { Console.WriteLine(Guid.NewGuid()); })); //Func<Task> Console.WriteLine(await Task.Run(() => Task.Run(() => Guid.NewGuid()))); //Func<Task<TResult>>
}
}
}

图2-3 Task.Run() 方法的重载

三、How 取消异步操作

  CancellationToken 和 CancellationTokenSource 这两个类允许你终止执行异步方法。

  (1)CancellationToken 对象包含任务是否被取消的信息;如果该对象的属性 IsCancellationRequested 为 true,任务需停止操作并返回;该对象操作是不可逆的,且只能使用(修改)一次,即该对象内的 IsCancellationRequested 属性被设置后,就不能改动。

  (2)CancellationTokenSource 可创建 CancellationToken 对象,调用 CancellationTokenSource 对象的 Cancel 方法,会使该对象的 CancellationToken 属性 IsCancellationRequested 设置为 true。

  【注意】调用 CancellationTokenSource 对象的 Cancel 方法,并不会执行取消操作,而是会将该对象的 CancellationToken 属性 IsCancellationRequested 设置为 true。

  示例

     internal class Program
{
private static void Main(string[] args)
{
CancellationTokenSource source = new CancellationTokenSource();
CancellationToken token = source.Token; var t = Do.ExecuteAsync(token); //Thread.Sleep(3000); //挂起 3 秒
//source.Cancel(); //传达取消请求 t.Wait(token); //等待任务执行完成
Console.WriteLine($"{nameof(token.IsCancellationRequested)}: {token.IsCancellationRequested}"); Console.Read();
} } internal class Do
{
/// <summary>
/// 异步执行
/// </summary>
/// <param name="token"></param>
/// <returns></returns>
public static async Task ExecuteAsync(CancellationToken token)
{
if (token.IsCancellationRequested)
{
return;
} await Task.Run(() => CircleOutput(token), token);
} /// <summary>
/// 循环输出
/// </summary>
/// <param name="token"></param>
private static void CircleOutput(CancellationToken token)
{
Console.WriteLine($"{nameof(CircleOutput)} 方法开始调用:"); const int num = ;
for (var i = ; i < num; i++)
{
if (token.IsCancellationRequested) //监控 CancellationToken
{
return;
} Console.WriteLine($"{i + 1}/{num} 完成");
Thread.Sleep();
}
}
}

图3-1

图3-2 注释两行代码

图3-3:图3-1和图3-2的执行结果(注释两行代码)

  上图是不调用 Cancel() 方法的结果图,不会取消任务的执行。

  下图在 3 秒后调用 Cancel() 方法取消任务的执行:

图3-4:去掉注释

图3-5:图3-1和图3-4的执行结果(去掉注释)

小结

  • 介绍异步方法的语法、三种不同的返回值类型(void、Task 和 Task<T>)和控制流程等。
  • 简单常用的异步执行方式:Task.Run()。【注意】它是在不同的线程上执行方法。
  • 如何取消异步操作。

传送门

  入门:《开始接触 async/await 异步编程

  补充篇:《走进异步编程的世界 - 剖析异步方法(下)

  GUI 篇:《走进异步编程的世界 - 在 GUI 中执行异步操作


原文链接:http://www.cnblogs.com/liqingwen/p/5844095.html

【参考】《Illustrated C# 2012》

[C#] 走进异步编程的世界 - 剖析异步方法(上)的更多相关文章

  1. [C#] 走进异步编程的世界 - 剖析异步方法(下)

    走进异步编程的世界 - 剖析异步方法(下) 序 感谢大家的支持,这是昨天发布<走进异步编程的世界 - 剖析异步方法(上)>的补充篇. 目录 异常处理 在调用方法中同步等待任务 在异步方法中 ...

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

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

  3. [C#] 走进异步编程的世界 - 在 GUI 中执行异步操作

    走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5877042.html 序 这是继<开始接 ...

  4. 走进异步编程的世界 - 开始接触 async/await(转)

    序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $"" 来拼接字符串,相当于string.Fo ...

  5. 走进异步编程的世界 - 在 GUI 中执行异步操作

    转载:https://www.cnblogs.com/liqingwen/p/5877042.html 走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://w ...

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

    原文链接:http://www.cnblogs.com/liqingwen/p/5831951.html 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 ...

  7. 走进异步编程的世界 - 开始接触 async/await

    [C#] 走进异步编程的世界 - 开始接触 async/await   走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async ...

  8. 走进异步编程的世界--async/await项目使用实战

    起因:今天要做一个定时器任务:五分钟查询一次数据库发现超时未支付的订单数据将其状态改为已经关闭(数据量大约100条的情况) 开始未使用异步: public void SelfCloseGpPayOrd ...

  9. javascript异步编程方案汇总剖析

    code[class*="language-"] { padding: .1em; border-radius: .3em; white-space: normal; backgr ...

随机推荐

  1. Android Studio配置 AndroidAnnotations——Hi_博客 Android App 开发笔记

    以前用Eclicps 用习惯了现在 想学学 用Android Studio 两天的钻研终于 在我电脑上装了一个Android Studio 并完成了AndroidAnnotations 的配置. An ...

  2. BootStrap_02之全局样式及组件

    1.BootStrap指定的四种屏幕尺寸: ①超大PC屏幕--lg(large):w>=1200px: ②中等PC屏幕--md(medium):1200px>w>=992px: ③P ...

  3. HTML DOM 介绍

    本篇主要介绍DOM内容.DOM 节点.节点属性以及获取HTML元素的方法. 目录 1. 介绍 DOM:介绍DOM,以及对DOM分类和功能的说明. 2. DOM 节点:介绍DOM节点分类和节点层次. 3 ...

  4. Node.js:理解stream

    Stream在node.js中是一个抽象的接口,基于EventEmitter,也是一种Buffer的高级封装,用来处理流数据.流模块便是提供各种API让我们可以很简单的使用Stream. 流分为四种类 ...

  5. 免费开源的.NET多类型文件解压缩组件SharpZipLib(.NET组件介绍之七)

    前面介绍了六种.NET组件,其中有一种组件是写文件的压缩和解压,现在介绍另一种文件的解压缩组件SharpZipLib.在这个组件介绍系列中,只为简单的介绍组件的背景和简单的应用,读者在阅读时可以结合官 ...

  6. 神技!微信小程序(应用号)抢先入门教程(附最新案例DEMO-豆瓣电影)持续更新

    微信小程序 Demo(豆瓣电影) 由于时间的关系,没有办法写一个完整的说明,后续配合一些视频资料,请持续关注 官方文档:https://mp.weixin.qq.com/debug/wxadoc/de ...

  7. C#各种同步方法 lock, Monitor,Mutex, Semaphore, Interlocked, ReaderWriterLock,AutoResetEvent, ManualResetEvent

    看下组织结构: System.Object System.MarshalByRefObject System.Threading.WaitHandle System.Threading.Mutex S ...

  8. Microsoft Visual Studio 2017 for Mac Preview 下载+安装+案例Demo

    目录: 0. 前言 1. 在线安装器 2. 安装VS 3. HelloWorld 4. ASP.NET MVC 5. 软件下载 6. 结尾 0. 前言: 工作原因,上下班背着我的雷神,一个月瘦了10斤 ...

  9. MySQL:Fabric 安装

    MySQL Fabric安装 MySQL Fabric是Oracle提供的用于辅助进行ha\sharding的工具,它的基本架构: 从上面看出,借助于Fabric, 可以搭建 HA 集群.Sharin ...

  10. IIS启动失败,启动Windows Process Activation Service时,出现错误13:数据无效 ;HTTP 错误 401.2 - Unauthorized 由于身份验证头无效,您无权查看此页

    因为修改过管理员账号的密码后重启服务器导致IIS无法启动,出现已下异常 1.解决:"启动Windows Process Activation Service时,出现错误13:数据无效&quo ...