一起谈.NET技术,.NET中的异步编程-Continuation passing style以及使用yield实现异步

简介:   传统的异步方式将本来紧凑的代码都分成两部分,不仅仅降低了代码的可读性,还让一些基本的程序构造无法使用,所以大部分开发人员在遇到应该使用异步的地方都忍痛割爱。本来我在本篇文章中想讨论一下.NET世界中已有的几个辅助异步开发的类库,但是经过思考后觉得在这之前介绍一下一些理论知识也许对理解后面的类库以及更新的内容有所帮助。

  传统的异步方式将本来紧凑的代码都分成两部分,不仅仅降低了代码的可读性,还让一些基本的程序构造无法使用,所以大部分开发人员在遇到应该使用异步的地方都忍痛割爱。本来我在本篇文章中想讨论一下.NET世界中已有的几个辅助异步开发的类库,但是经过思考后觉得在这之前介绍一下一些理论知识也许对理解后面的类库以及更新的内容有所帮助。今天我们要讨论的是Continuation Passing Style,简称CPS。

  CPS

  首先,我们看看下面这个方法:

   1: public int Add(int a, int b)
   2: {
   3:     return a + b;
   4: }

  我们一般这样调用它:

   1: Print(Add(5, 6))
   2:  
   3: public void Print(int result)
   4: {
   5:     Console.WriteLine(result);
   6: }

  如果我们以CPS的方式编写上面的代码则是这个样子:

   1: public void Add(int a, int b, Action<int> continueWith)
   2: {
   3:     continueWith(a+b);
   4: }
   5:  
   6: Add(5, 6, (ret) => Print(ret));

  就好像我们将方法倒过来,我们不再是直接返回方法的结果;我们现在做的是接受一个委托,这个委托表示我这个方法运算完后要干什么,就是传说的continue。对于这里来说,Add的continue就是Print。

不仅是上面这样的代码示例。在一个方法中,在本语句后面执行的语句都可以称之为本语句的continue。

  CPS 与 Async

  那么可能有人要问,你说这么多跟异步有什么关系么?对,跟异步有很大的关系。回想上一篇文章,经典的异步模式都是一个以Begin开头的方法发起异步请求,并且向这个方法传入一个回调(callback),当异步执行完毕后该回调会被执行,那么我们可以称该回调为这个异步请求的continue:

   1: stream.BeginRead(buffer, 0, 1024, continueWith, null)

  这又有什么用呢?那先来看看我们期望写出什么样子的异步代码吧(注意,这是伪代码,不要没有看文章就直接粘贴代码到vs运行):

   1: var request = HttpWebRequest.Create("http://www.google.com");
   2: var asyncResult1 = request.BeginGetResponse(...);
   3: var response = request.EndGetResponse(asyncResult1);
   4: using(stream = response.GetResponseStream())
   5: {
   6:     var asyncResult2 = stream.BeginRead(buffer, 0, 1024, ...);
   7:     var actualRead = stream.EndRead(asyncResult2);
   8: }

  对,我们想要像同步的方式一样编写异步代码,我讨厌那么多回调,特别是一环嵌套一环的回调。

  参照前面对CPS的讨论,在request.BeginGetResponse之后的代码,都是它的continue,如果我能够有一种机制获得我的continue,然后在我执行完毕之后调用continue该多好啊。可惜,C#没有像Scheme那样的控制操作符call/cc获取continue。

  思路貌似到这儿断了。但是我们是否可以换个角度想想,如果我们能给上面这段代码加上标识:在每个异步请求发起的地方都加一个标识,而标识之后的部分就是continue。

var request = HttpWebRequest.Create("http://www.google.com");
标识1 var asyncResult1 = request.BeginGetResponse(...);
var response = request.EndGetResponse(asyncResult1);
using(stream = response.GetResponseStream())
{
    标识2 var asyncResult2 = stream.BeginRead(buffer, 0, 1024, ...);
    var actualRead = stream.EndRead(asyncResult2);
}

  当执行到 标识1 时,立即返回,并且记住本次执行只执行到了 标识1,当异步请求完毕后,它知道上次执行到了 标识1,那么这个时候就从标识1的下一行开始执行,当执行到标识2时,又遇到一个异步请求,立即返回并记住本次执行到了标识2,然后请求完毕后从标识2的下一行恢复执行。那么现在的任务就是如果打标识以及在异步请求完毕后如何从标识位置开始恢复执行。

  yield 与 异步

  如果你熟悉C# 2.0加入的迭代器特性,你就会发现yield就是我们可以用来打标识的东西。看下面的代码:

   1: public IEnumerator<int> Demo()
   2: {
   3:     //code 1
   4:     yield return 1;
   5:     //code 2
   6:     yield return 2;
   7:     //code 3
   8:     yield return 3;
   9: }

  经过编译会生成类似下面的代码(伪代码,相差很远,只是意义相近,想要了解详情的同学可以自行打开Reflector观看):

   1: public IEnumerator<int> Demo()
   2: {
   3:    return new GeneratedEnumerator();
   4: }
   5:  
   6: public class GeneratedEnumerator
   7: {
   8:     private int state = 0;
   9:  
  10:     private int currentValue = 0;
  11:     
  12:     public bool MoveNext()
  13:     {
  14:         switch(state)
  15:         {
  16:             case 0:
  17:                 //code 1
  18:                 currentValue = 1;
  19:                 state = 1;
  20:                 return true;
  21:             case 1:
  22:                 //code 2
  23:                 currentValue = 2;
  24:                 state = 2;
  25:                 return true;
  26:             case 2:
  27:                 //code 3
  28:                 currentValue = 3;
  29:                 state = 3;
  30:                 return true;
  31:             default:return false;
  32:         }
  33:     }
  34:     
  35:     public int Current{get{return currentValue;}}
  36: }

  对,C#编译器将其翻译成了一个状态机。yield return就好像做了很多标记,MoveNext每调用一次,它就执行下个yield return之前的代码,然后立即返回。

  好,现在打标记的功能有了,我们如何在异步请求执行完毕后恢复调用呢?通过上面的代码,你可能已经想到了,我们这里恢复调用只需要再次调用一下MoveNext就行了,那个状态机会帮我们处理一切。

  那我们改造我们的异步代码:

   1: public IEnumerator<int> Download()
   2: {
   3:     var request = HttpWebRequest.Create("http://www.google.com");
   4:     var asyncResult1 = request.BeginGetResponse(...);
   5:     yield return 1;
   6:     var response = request.EndGetResponse(asyncResult1);
   7:     using(stream = response.GetResponseStream())
   8:     {
   9:         var asyncResult2 = stream.BeginRead(buffer, 0, 1024, ...);
  10:         yield return 1;
  11:         var actualRead = stream.EndRead(asyncResult2);
  12:     }
  13: }

  标记打好了,考虑如何在异步调用完执行一下MoveNext吧。

  呵呵,你还记得异步调用的那个AsyncCallback回调么?也就是异步请求执行完会调用的那个。如果我们向发起异步请求的BeginXXX方法传入一个AsyncCallback,而这个回调里会调用MoveNext怎么样?

   1: public IEnumerator<int> Download(Context context)
   2: {
   3:     var request = HttpWebRequest.Create("http://www.google.com");
   4:     var asyncResult1 = request.BeginGetResponse(context.Continue(),null);
   5:     yield return 1;
   6:     var response = request.EndGetResponse(asyncResult1);
   7:     using(stream = response.GetResponseStream())
   8:     {
   9:         var asyncResult2 = stream.BeginRead(buffer, 0, 1024, context.Continue(),null);
  10:         yield return 1;
  11:         var actualRead = stream.EndRead(asyncResult2);
  12:     }
  13: }

  Continue方法的定义是:

   1: public class Context
   2: {
   3:     //...
   4:     private IEnumerator enumerator;
   5:     
   6:     public AsyncCallback Continue()
   7:     {
   8:         return (ar) => enumerator.MoveNext();
   9:     }
  10: }

  在调用Continue方法之前,Context类还必须保存有Download方法返回的IEnumerator,所以:

   1: public class Context
   2: {
   3:     //...
   4:     private IEnumerator enumerator;
   5:     
   6:     public AsyncCallback Continue()
   7:     {
   8:         return (ar) => enumerator.MoveNext();
   9:     }
  10:  
  11:     public void Run(IEnumerator enumerator)
  12:     {
  13:         this.enumerator = enumerator;
  14:         enumerator.MoveNext();
  15:     }
  16: }

  那调用Download的方法就可以写成:

   1: public void Main()
   2: {
   3:     Program p = new Program();
   4:     
   5:     Context context = new Context();
   6:     context.Run(p.Download(context));
   7: }

  除了执行方式的不同外,我们几乎就可以像同步的方式那样编写异步的代码了。

  完整的代码如下(为了更好的演示,我将下面代码改为Winform版本):

   1: public class Context
   2: {
   3:     private IEnumerator enumerator;
   4:     
   5:     public AsyncCallback Continue()
   6:     {
   7:         return (ar) => enumerator.MoveNext();
   8:     }
   9:  
  10:     public void Run(IEnumerator enumerator)
  11:     {
  12:         this.enumerator = enumerator;
  13:         enumerator.MoveNext();
  14:     }
  15: }
  16:  
  17: private void btnDownload_click(
目录
相关文章
|
5月前
|
监控 Cloud Native 测试技术
.NET技术深度解析:现代企业级开发指南
每日激励:“不要一直责怪过去的自己,他曾经站在雾里也很迷茫”。我是蒋星熠Jaxonic,一名在代码宇宙中探索的极客旅人。从.NET Framework到.NET 8,我深耕跨平台、高性能、云原生开发,践行领域驱动设计与微服务架构,用代码书写技术诗篇。分享架构演进、性能优化与AI融合前沿,助力开发者在二进制星河中逐光前行。关注我,共探技术无限可能!
.NET技术深度解析:现代企业级开发指南
|
11月前
|
SQL 小程序 API
如何运用C#.NET技术快速开发一套掌上医院系统?
本方案基于C#.NET技术快速构建掌上医院系统,结合模块化开发理念与医院信息化需求。核心功能涵盖用户端的预约挂号、在线问诊、报告查询等,以及管理端的排班管理和数据统计。采用.NET Core Web API与uni-app实现前后端分离,支持跨平台小程序开发。数据库选用SQL Server 2012,并通过读写分离与索引优化提升性能。部署方案包括Windows Server与负载均衡设计,确保高可用性。同时针对API差异、数据库老化及高并发等问题制定应对措施,保障系统稳定运行。推荐使用Postman、Redgate等工具辅助开发,提升效率与质量。
470 0
|
开发框架 算法 .NET
C#/.NET/.NET Core技术前沿周刊 | 第 15 期(2024年11.25-11.30)
C#/.NET/.NET Core技术前沿周刊 | 第 15 期(2024年11.25-11.30)
256 6
|
开发框架 Cloud Native .NET
C#/.NET/.NET Core技术前沿周刊 | 第 16 期(2024年12.01-12.08)
C#/.NET/.NET Core技术前沿周刊 | 第 16 期(2024年12.01-12.08)
278 6
|
开发框架 安全 .NET
在数字化时代,.NET 技术凭借跨平台兼容性、丰富的开发工具和框架、高效的性能及强大的安全稳定性,成为软件开发的重要支柱
在数字化时代,.NET 技术凭借跨平台兼容性、丰富的开发工具和框架、高效的性能及强大的安全稳定性,成为软件开发的重要支柱。它不仅加速了应用开发进程,提升了开发质量和可靠性,还促进了创新和业务发展,培养了专业人才和技术社区,为软件开发和数字化转型做出了重要贡献。
368 5
|
传感器 人工智能 供应链
.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。
本文深入探讨了.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。通过企业级应用、Web应用及移动应用的创新案例,展示了.NET在各领域的广泛应用和巨大潜力。展望未来,.NET将与新兴技术深度融合,拓展跨平台开发,推动云原生应用发展,持续创新。
199 4
|
开发框架 .NET C#
.NET 技术凭借高效开发环境、强大框架支持及跨平台特性,在软件开发中占据重要地位
.NET 技术凭借高效开发环境、强大框架支持及跨平台特性,在软件开发中占据重要地位。从企业应用到电子商务,再到移动开发,.NET 均展现出卓越性能,助力开发者提升效率与项目质量,推动行业持续发展。
408 4
|
机器学习/深度学习 人工智能 Cloud Native
在数字化时代,.NET 技术凭借其跨平台兼容性、丰富的类库和工具集以及卓越的性能与效率,成为软件开发的重要平台
在数字化时代,.NET 技术凭借其跨平台兼容性、丰富的类库和工具集以及卓越的性能与效率,成为软件开发的重要平台。本文深入解析 .NET 的核心优势,探讨其在企业级应用、Web 开发及移动应用等领域的应用案例,并展望未来在人工智能、云原生等方面的发展趋势。
349 3
|
敏捷开发 缓存 中间件
.NET技术的高效开发模式,涵盖面向对象编程、良好架构设计及高效代码编写与管理三大关键要素
本文深入探讨了.NET技术的高效开发模式,涵盖面向对象编程、良好架构设计及高效代码编写与管理三大关键要素,并通过企业级应用和Web应用开发的实践案例,展示了如何在实际项目中应用这些模式,旨在为开发者提供有益的参考和指导。
167 3
|
开发框架 安全 Java
.NET技术的独特魅力与优势,涵盖高效的开发体验、强大的性能表现、高度的可扩展性及丰富的生态系统等方面,展示了其在软件开发领域的核心竞争力
本文深入探讨了.NET技术的独特魅力与优势,涵盖高效的开发体验、强大的性能表现、高度的可扩展性及丰富的生态系统等方面,展示了其在软件开发领域的核心竞争力。.NET不仅支持跨平台开发,具备出色的安全性和稳定性,还能与多种技术无缝集成,为企业级应用提供全面支持。
496 3