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

  这是学习异步编程的入门篇。

  涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $"" 来拼接字符串,相当于string.Format() 方法。

 

目录

  • What's 异步?

  • async/await 结构

  • What’s 异步方法?

 

一、What's 异步?

     启动程序时,系统会在内存中创建一个新的进程。进程是构成运行程序资源的集合。

     在进程内部,有称为线程的内核对象,它代表的是真正的执行程序。系统会在 Main 方法的第一行语句就开始线程的执行。

 

     线程

     ①默认情况,一个进程只包含一个线程,从程序的开始到执行结束;

     ②线程可以派生自其它线程,所以一个进程可以包含不同状态的多个线程,来执行程序的不同部分;

     ③一个进程中的多个线程,将共享该进程的资源;

     ④系统为处理器执行所规划的单元是线程,而非进程

 

     一般来说我们写的控制台程序都只使用了一个线程,从第一条语句按顺序执行到最后一条。但在很多的情况下,这种简单的模型会在性能或用户体验上不好。

     例如:服务器要同时处理来自多个客户端程序的请求,又要等待数据库和其它设备的响应,这将严重影响性能。程序不应该将时间浪费在响应上,而要在等待的同时执行其它任务!

     现在我们开始进入异步编程。在异步程序中,代码不需要按照编写时的顺序执行。这时我们需要用到 C# 5.0 引入的 async/await 来构建异步方法。

 

     我们先看一下不用异步的示例:

class Program

    {

        //创建计时器

        private static readonly Stopwatch Watch = new Stopwatch();


        private static void Main(string[] args)

        {

            //启动计时器

            Watch.Start();


            const string url1 = "http://www.cnblogs.com/";

            const string url2 = "http://www.cnblogs.com/liqingwen/";


            //两次调用 CountCharacters 方法(下载某网站内容,并统计字符的个数)

            var result1 = CountCharacters(1, url1);

            var result2 = CountCharacters(2, url2);


            //三次调用 ExtraOperation 方法(主要是通过拼接字符串达到耗时操作)

            for (var i = 0; i < 3; i++)

            {

                ExtraOperation(i + 1);

            }


            //控制台输出

            Console.WriteLine($"{url1} 的字符个数:{result1}");

            Console.WriteLine($"{url2} 的字符个数:{result2}");


            Console.Read();

        }


        /// <summary>

        /// 统计字符个数

        /// </summary>

        /// <param name="id"></param>

        /// <param name="address"></param>

        /// <returns></returns>

        private static int CountCharacters(int id, string address)

        {

            var wc = new WebClient();

            Console.WriteLine($"开始调用 id = {id}:{Watch.ElapsedMilliseconds} ms");


            var result = wc.DownloadString(address);

            Console.WriteLine($"调用完成 id = {id}:{Watch.ElapsedMilliseconds} ms");


            return result.Length;

        }


        /// <summary>

        /// 额外操作

        /// </summary>

        /// <param name="id"></param>

        private static void ExtraOperation(int id)

        {

            //这里是通过拼接字符串进行一些相对耗时的操作

            var s = "";


            for (var i = 0; i < 6000; i++)

            {

                s += i;

            }


            Console.WriteLine($"id = {id} 的 ExtraOperation 方法完成:{Watch.ElapsedMilliseconds} ms");

        }

    }

     图1-1 运行的效果图,以毫秒(ms)为单位

 

  【备注】一般来说,直接拼接字符串是一种比较耗性能的手段,如果对字符串拼接有性能要求的话应该使用 StringBuilder。

  【注意】每次运行的结果可能不同。不管哪次调试,绝大部分时间都浪费前两次调用(CountCharacters 方法),即在等待网站的响应上。

 

  图1-2 根据执行结果所画的时间轴

 

     有人曾幻想着这样提高性能的方法:在调用 A 方法时,不等它执行完,直接执行 B 方法,然后等 A 方法执行完成再处理。

     C# 的 async/await 就可以允许我们这么弄。

 这是修改后的代码

 图1-3 修改后的执行结果图

图1-4 根据加入异步后的执行结果画的时间轴。

 

  我们观察时间轴发现,新版代码比旧版快了不少(由于网络波动的原因,很可能会出现耗时比之前长的情况)。这是由于 ExtraOperation 方法的数次调用是在 CountCharactersAsync 方法调用时等待响应的过程中进行的。所有的工作都是在主线程中完成的,没有创建新的线程。

 

  【改动分析】只改了几个细节的地方,直接展开代码的话可能看不出来,改动如下:

   

 图1-5

  图1-6

 

  ①从 Main 方法执行到 CountCharactersAsync(1, url1) 方法时,该方法会立即返回,然后才会调用它内部的方法开始下载内容。该方法返回的是一个 Task<int> 类型的占位符对象,表示计划进行的工作。这个占位符最终会返回 int 类型的值。

  ②这样就可以不必等 CountCharactersAsync(1, url1) 方法执行完成就可以继续进行下一步操作。到执行 CountCharactersAsync(2, url2)  方法时,跟 ① 一样返回 Task<int> 对象。

  ③然后,Main 方法继续执行三次 ExtraOperation 方法,同时两次 CountCharactersAsync 方法依然在持续工作 。

  ④t1.Result 和 t2.Result 是指从 CountCharactersAsync 方法调用的 Task<int> 对象取结果,如果还没有结果的话,将阻塞,直有结果返回为止。

 

二、async/await 结构

     先解析一下专业名词:

     同步方法:一个程序调用某个方法,等到其执行完成之后才进行下一步操作。这也是默认的形式。

     异步方法:一个程序调用某个方法,在处理完成之前就返回该方法。通过 async/await 我们就可以实现这种类型的方法。

 

     async/await 结构可分成三部分:

     (1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;

     (2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;

     (3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。

 

  现在我们来分析一下示例。

  图2-1

 

 三、What’s 异步方法

     异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。

     语法分析:

     (1)关键字:方法头使用 async 修饰。

     (2)要求:包含 N(N>0) 个 await 表达式(不存在 await 表达式的话 IDE 会发出警告),表示需要异步执行的任务。

     (3)返回类型:只能返回 3 种类型(void、Task 和 Task<T>)。Task 和 Task<T> 标识返回的对象会在将来完成工作,表示调用方法和异步方法可以继续执行。

     (4)参数:数量不限,但不能使用 out 和 ref 关键字。

     (5)命名约定:方法后缀名应以 Async 结尾。

     (6)其它:匿名方法和 Lambda 表达式也可以作为异步对象;async 是一个上下文关键字;关键字 async 必须在返回类型前。

 

图3-1 异步方法的简单结构图

 

小结

  1.解析了进程和线程的概念

  2.异步的简单用法

  3.async/await 结构体

  4.异步方法语法结构

相关文章: 

  • Async/Await 异步编程中的最佳做法

  • [C#]async和await刨根问底

  • 剖析异步编程语法糖: async和await

  • Async/Await 异步编程中的最佳做法

原文地址:http://www.cnblogs.com/liqingwen/p/5831951.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/328527.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

所有的软弱,都是昂贵的

作者&#xff1a;周冲 来源&#xff1a;周冲的影像声色&#xff08;fuck_your_dick&#xff09; 01 契诃夫有一个小说&#xff0c;名叫《柔弱的人》&#xff0c;讲一个主人&#xff0c;巧立名目&#xff0c;剥夺家庭教师的工资。 首先&#xff0c;他压缩掉她的周末&#xff0c…

python3中urlopen_解决python3 urllib中urlopen报错的问题

前言最近更新了Python版本&#xff0c;准备写个爬虫&#xff0c;意外的发现urllib库中属性不存在urlopen&#xff0c;于是各种google&#xff0c;然后总结一下给出解决方案问题的出现AttributeError: module object has no attribute urlopen问题的解决途径我们先来看下官方文档…

100+经典Java面试题及答案解析

转载自 100经典Java面试题及答案解析面向对象编程&#xff08;OOP&#xff09;Java是一个支持并发、基于类和面向对象的计算机编程语言。下面列出了面向对象软件开发的优点&#xff1a; 代码开发模块化&#xff0c;更易维护和修改。代码复用。增强代码的可靠性和灵活性。增加代…

.NET HttpClient的缺陷和文档错误让开发人员倍感沮丧

设计错误、缺陷及文档错误等导致正确使用.NET HttpClient变得出奇地困难。所以&#xff0c;即使是生产环境中看似运行正常的应用程序&#xff0c;在负荷不满的情况下&#xff0c;也遭受着性能问题和运行时故障。 来自ASP.NET Monsters的Simon Timms就通过一篇题为“你正在错误地…

python监听剪贴板_Python监听剪切板实现方法代码实例

第一种import win32clipboardimport time#速度快 容易出错class niubi():def lihai(self):while True:#jianting().main()t jianting().main()print(t)class jianting():def clipboard_get(self):"""获取剪贴板数据"""win32clipboard.OpenClipb…

新闻发布项目——业务逻辑层(UserService)

package bdqn.newsManageServlet.Service;import bdqn.newsManageServlet.entity.User;/*** 用户信息* author Administrator**/ public interface UserService {public User getLoginUser(String uName,String pwd, String msg);}

二叉树总结挺好的很好记忆

https://blog.csdn.net/fightforyourdream/article/details/16843303 面试大总结之二&#xff1a;Java搞定面试中的二叉树题目 2013年11月20日 14:04:27 chiiis 阅读数&#xff1a;25438更多 个人分类&#xff1a; AlgorithmInterview 这是本系列的第二篇&#xff0c;与前一…

免费开源分布式系统日志收集框架 Exceptionless

前言 从去年就答应过Eric&#xff08;Exceptionless的作者之一&#xff09;&#xff0c;在中国会帮助给 Exceptionless 做推广&#xff0c;但是由于各种原因一直没有做这件事情&#xff0c;在此对Eric表示歉意。:) Exceptionless 简介 Exceptionless 是一个开源的实时的日志收集…

如何安装mysql5.5.6_centos6安装mysql5.5.53

MysqL官网下载MysqL的red hat linux安装包下载地址为&#xff1a;http://dev.MysqL.com/downloads/MysqL/5.5.html#downloads下载后的文件为&#xff1a;MysqL-5.5.53-1.el7.x86_64.rpm-bundle.tar解压 tar 包 &#xff1a;tar -xvf MysqL-5.5.53-1.el7.x86_64.rpm-bundle.tar移…

java职业规划

百度搜索java职业规划好多啊啊啊

避免同步死锁

翻译自 避免同步死锁在我之前的文章“ Double-Checked Locking&#xff1a;Clever&#xff0c;but Broken ”&#xff08;JavaWorld&#xff0c;2001年2月&#xff09;&#xff0c;我描述了几种常用的避免同步的技术实际上是不安全的&#xff0c;并建议了一个“如有疑问&#…

程序员小测试:保守派 vs 自由派

最近&#xff0c;我在阅读 Steve Yegg 的文集《程序员的呐喊》。 这是一本非常有趣的书&#xff0c;里面甚至包含了一个小测试&#xff08;原文&#xff09;&#xff0c;区分一个程序员到底是保守派还是自由派。 下面一共有十个问题&#xff0c;每个问题都有 A 和 B 两个选项&a…

新闻发布项目——业务逻辑层(newsTbService)

package bdqn.newsManageServlet.Service;import java.util.List;import bdqn.newsManageServlet.entity.newsTb;/*** 新闻业务逻辑层的接口* author Administrator**/ public interface newsTbService {//分页查询public List<newsTb>getPagingNews(int pagesize,int pa…

django mysql 创建表_关于 django ORM 中,数据库建表方式的问题

本人以前是做客户端的&#xff0c;做后端没多久。一直有一个问题困扰我&#xff0c;正好现在手上有一个系统在做。系统逻辑是写一个爬虫在网上爬取一个网站的信息&#xff0c;经过加工处理后&#xff0c;使用 django 来展示和交互。表结构简化为三张表A:爬虫爬取得到的信息B:数…

遍历多叉树

https://www.jianshu.com/p/dee8284b2dc4 beg4 关注 2018.03.22 15:14* 字数 334 阅读 172评论 0喜欢 1 随便画一个树,写代码遍历它 OK,树的结构这么描述 public class TreeNode {private String name;private TreeNode parent;private List<TreeNode> children new …

tomcat与apache的面试题

转载自 tomcat与apache的面试题tomcat与apache有哪几种连接方式&#xff1f; Tomcat 与 Apache 有三种连接方式&#xff0c; 1、JK方式 这是最常见的方式。JK 是通过 AJP 协议与 Tomcat 服务器进行通讯的&#xff0c;Tomcat 默认的 AJP Connector 的端口是 8009。JK 本身提供…

使用 Exceptionless 作为 Log Server 搭配 NLog 记录系统日志

昨天的文章<免费开源分布式系统日志收集框架 Exceptionless>反响很大,今天推的一篇是续集,文章来自于宝岛台湾的MVP, 让 Exceptionless 建置 Log Server,性能不用担心,用的是大名点点的ELK组合. 前言 痾...久违的新文章&#xff0c;让我的眼角流下了蛋蛋忧伤的泪珠.... 今…

elementui下拉框选择图片_element ui下拉框如何实现默认选择?

为什么我这样写没反应呢运营商级别 <el-optionv-for"item in options1":label"item.label":value"item.value"></el-option>export default{data(){return{selected:,options1:[{value: 选项1,label: 省级运营商}, {value: 选项2,la…

程序员成长之路 java面试指导(作者说的极好要看) 静下心看

https://blog.csdn.net/weixin_41780944/article/details/79429769 从面试官的角度分享一些后端校招经验&#xff08;作者说的极好要看&#xff09; https://blog.csdn.net/qq_34337272/article/details/80875016 除了手写代码和计算机基础&#xff0c;剩下的方面都是可选项…

[MySQL] 二进制的应用场景

前言 MySQL 是一种常用的关系型数据库管理系统&#xff0c;广泛应用于各种软件和网站开发中。在 MySQL 中&#xff0c;数据以二进制文件的形式存储在硬盘上。这些二进制文件不仅可以提供数据的持久化存储&#xff0c;还可以在不同的应用场景中发挥重要作用。本文将介绍 MySQL …