async/await Task.Delay 和Thread.Sleep的理解

相关学习资料:

第十七节:从状态机的角度async和await的实现原理(新) - Yaopengfei - 博客园 (cnblogs.com)

[基础知识]有限状态机_哔哩哔哩_bilibili

C# async await 原理:编译器如何将异步函数转换成状态机 | 码农网 (codercto.com)

await——调用的等待期间,.NET会把当前的线程返回给线程池,等异步方法调用执行完毕后,框架会从线程池再取出来一个线程执行后续的代码,当前线程不会阻塞

从原理层面刨析

  1. async与await是语法糖,最终译成“状态机调用”。其他常用的语法糖var、using、lambda表达式等。

  2. async关键字标记的方法会被C#编译器编译成一个状态机。

  3. await是关键字是为了实现状态机中的一个状态。

    会根据方法内的await调用切分成多个状态,每当有一个await,就会生成一个对应的状态。

  4. 状态机实现了AsyncStateMachine接口,里面有MoveNext 和 SetStateMachine方法处理相应业务.

    MoveNext——定义各个状态之间转换的方法

    1. 在await执行时调用一次,在await操作结束时继续调用

从实践方向刨析

  1. async会将方法的返回结果包装成Task,所以它不是必需的

            //使用async,实际是将Result包装成Task<Result>
    //此处就是对结果通过await拆包,再通过async包装成Task<Result>返回,当前场景下没有必要,可直接简写成下面的场景
    public static async Task AsyncNoReturn()
    {
    await Task.Delay(500);
    }
    //不使用async的写法,因为类型已经是Task了不需要通过async关键字包装
    public static Task AsyncNoReturn1()
    {
    return Task.Delay(500);
    }

    总结:如果一个异步方法只是对别的异步方法调用的转发,并没有太多复杂的逻辑(比如等待的结果,再调用B:把调用的返回值拿到内部做一些处理再返回),那么就可以去掉async关键字。返回值为Task的方法不一定都要标注async,标注async只是让我们可以更方便的await而已

    1. 避免对返回结果Task的“拆包后再次包装”
    2. 避免在底层创建状态机,性能更优

    正面Demo:

            public static Task<string> DownloadFromFile(int num)
    {
    if (num == 1)
    {
    return File.ReadAllTextAsync(@"D:\1.txt");
    }
    else if (num == 2)
    {
    return File.ReadAllTextAsync(@"D:\2.txt");
    }
    else
    {
    throw new ArgumentException("Invalid Number");
    }
    }

    如果没有在声明Task的时候前面没有加await,则主线程不会异步等待Task完成,主线程会继续往下执行,与Task并发执行

    Task.Delay() 和 Thread.Sleep() 区别

Thread.Sleep() ——会让当前线程休眠,形成阻塞,当前休眠结束后继续往下执行

await Task.Delay()——当前线程返回线程池,从线程池拿另外一个空闲线程线程做定时任务,等待任务结束后,从线程池拿到一个空闲线程,继续往下执行。

            await Task.Delay(200);
//可理解等效成如下代码
await Task.Run(() =>
{
Thread.Sleep(200);
});

区别:

  1. Thread.Sleep会阻塞当前线程,但是从线程池另取线程;Task.Delay不会阻塞当前线程,但是会新取一个线程

  2. Thread.Sleep不可取消,Task.Delay可取消

            public static Task Test_Delay()
    {
    //创建一个5秒的异步等待
    Task delay1 = Task.Delay(TimeSpan.FromSeconds(5));
    return delay1;
    }

    在Main方法里测试

    static void Main(string[] args)
    {
    //Task在声明处就开始执行
    var test = Test_Delay();
    //主线程等待2秒钟
    Thread.Sleep(TimeSpan.FromSeconds(2));
    Stopwatch sw = new Stopwatch();
    sw.Start();
    //当前线程等待test指向的Task执行结束
    test.Wait();
    sw.Stop();
    TimeSpan ts = sw.Elapsed;
    Console.WriteLine($"Task.Delay执行结束,耗时:{ts.TotalMilliseconds}");
    Console.ReadKey();
    }

    结果打印:

    Task.Delay执行结束,耗时:3004.527

    上面显示大概监听为3秒。这个表示了当前主线程在运行的时候,Task.Delay也在运行,这个只有在不同线程才可以实现。

async/await Task.Delay 和Thread.Sleep的理解的更多相关文章

  1. async/await Task Timeout

    async/await Task Timeout 在日常的电脑使用过程中,估计最难以忍受的就是软件界面"卡住""无响应",在我有限的开发生涯中一直都是在挑战 它 ...

  2. Task C# 多线程和异步模型 TPL模型 【C#】43. TPL基础——Task初步 22 C# 第十八章 TPL 并行编程 TPL 和传统 .NET 异步编程一 Task.Delay() 和 Thread.Sleep() 区别

    Task C# 多线程和异步模型 TPL模型   Task,异步,多线程简单总结 1,如何把一个异步封装为Task异步 Task.Factory.FromAsync 对老的一些异步模型封装为Task ...

  3. async await Task

    一.使用Task 引用命名空间 using System.Threading.Tasks; 1.工厂方式 Task.Factory.StartNew(() => {Console.WriteLi ...

  4. async,await,Task 的一些用法

    async,await,Task 的一些用法 private void Form1_Load(object sender, EventArgs e) { Display(); } public asy ...

  5. C#扫盲篇(四):.NET Core 的异步编程-只讲干货(async,await,Task)

    关于async,await,task的用法和解释这里就不要说明了,网上一查一大堆.至于为啥还要写这篇文章,主要是其他文章水分太多,不适合新手学习和理解.以下内容纯属个人理解,如果有误,请高手指正.本文 ...

  6. c#5.0(.net 4.5之后)的 Async+await+Task的异步机制的调试笔记

    1.)无返回值的情况(异步也是基于线程). using System; using System.Collections.Generic; using System.Linq; using Syste ...

  7. Task.Delay() 和 Thread.Sleep() 区别

    1.Thread.Sleep 是同步延迟,Task.Delay异步延迟. 2.Thread.Sleep 会阻塞线程,Task.Delay不会. 3.Thread.Sleep不能取消,Task.Dela ...

  8. task.delay 和 thread.sleep

    1.Thread.Sleep 是同步延迟. Task.Delay异步延迟. 2.Thread.Sleep 会阻塞线程,Task.Delay不会. 3.Thread.Sleep不能取消,Task.Del ...

  9. async await Task 使用方法

    使用概述 C#的使用过程中,除了以前的Thread.ThreadPool等用来开一个线程用来处理异步的内容.还可以使用新特性来处理异步.比以前的Thread和AutoResetEvent.delege ...

  10. async await task.Result 卡死

    在如下代码中: public async Task<string> GetData() { return await DoWork(); } 在UI线程中调用 var data = Get ...

随机推荐

  1. RocketMQ实战—6.生产优化及运维方案

    大纲 1.RocketMQ集群如何进行权限机制的控制 2.如何对RocketMQ集群进行消息堆积的追踪 3.如何处理RocketMQ的百万消息积压问题 4.针对RocketMQ集群崩溃的金融级高可用方 ...

  2. [业界方案] Yarn的业界解决方案和未来方向

    [业界方案] Yarn的业界解决方案和未来方向 目录 [业界方案] Yarn的业界解决方案和未来方向 0x00 摘要 0x01 Yarn 1.1 参考文章 0x02 分析 2.1 综述 2.1.1 y ...

  3. Q:记录一次ssh毫无规律的断线

    查找tailf /var/log/secure 网上一般是这三个原因 1.ssh服务端配置 vim /etc/ssh/sshd_config #客户端每隔多少秒向服务发送一个心跳数据,0代表不发送#C ...

  4. ORACLE11g数据中创建DB Link方法,用于跨oracle数据库查询数据

    ---查看该用户下已建立的DB link链接 SELECT * FROM DBA_DB_LINKS --创建语句 CREATE DATABASE LINK  连接名CONNECT TO 登录名  ID ...

  5. ORACLE SELECT INTO 赋值为空,抛出 NO DATA FOUND 异常

    例子: DECLARE ORDER_NUM VARCHAR2(20); BEGIN SELECT S.ORDER_NUM INTO ORDER_NUM FROM SALES_ORDER S WHERE ...

  6. 小程序开发实战案例五 | 小程序如何嵌入H5页面

    在接入小程序过程中会遇到需要将 H5 页面集成到小程序中情况,今天我们就来聊一聊怎么把 H5 页面塞到小程序中. 本篇文章将会从下面这几个方面来介绍: 小程序承载页面的前期准备 小程序如何承载 H5 ...

  7. .NET Core常用集合的几个坑

    C#中的常见集合 注意,箭头线不代表继承关系,只代表功能上的加强,如有错误,欢迎指出. 泛型集合时间复杂度 集合类型 添加 删除 查找 访问(索引/键) 遍历 备注 List<T> O(1 ...

  8. Android开发之定时任务(AlarmManager、WorkManager)

    Android 程序的定时任务主要有AlarmManager.WorkManager两种. 一.AlarmManager AlarmManager,又称闹钟,可以设置一次性任务,周期重复任务,定时重复 ...

  9. CentOS7安装部署ClickHouse(单机版&&集群部署)

    1.1 什么是ClickHouse ClickHouse 是俄罗斯的Yandex于2016年开源的列式存储数据库(DBMS),主要用于在线分析处理查询(OLAP),能够使用SQL查询实时生成分析数据报 ...

  10. google浏览器删除token

    测试登录时长,页面是否返回到首页 删除token