async/await Task.Delay 和Thread.Sleep的理解
async/await Task.Delay 和Thread.Sleep的理解
相关学习资料:
第十七节:从状态机的角度async和await的实现原理(新) - Yaopengfei - 博客园 (cnblogs.com)
C# async await 原理:编译器如何将异步函数转换成状态机 | 码农网 (codercto.com)
await——调用的等待期间,.NET会把当前的线程返回给线程池,等异步方法调用执行完毕后,框架会从线程池再取出来一个线程执行后续的代码,当前线程不会阻塞
从原理层面刨析
async与await是
语法糖,最终译成“状态机调用”。其他常用的语法糖var、using、lambda表达式等。async关键字标记的方法会被C#编译器编译成一个状态机。
await是关键字是为了实现状态机中的一个状态。
会根据方法内的await调用切分成多个状态,每当有一个await,就会生成一个对应的状态。
状态机实现了AsyncStateMachine接口,里面有MoveNext 和 SetStateMachine方法处理相应业务.

MoveNext——定义各个状态之间转换的方法
- 在await执行时调用一次,在await操作结束时继续调用
从实践方向刨析
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而已
- 避免对返回结果Task的“拆包后再次包装”
- 避免在底层创建状态机,性能更优
正面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并发执行
Thread.Sleep() ——会让当前线程休眠,形成阻塞,当前休眠结束后继续往下执行
await Task.Delay()——当前线程返回线程池,从线程池拿另外一个空闲线程线程做定时任务,等待任务结束后,从线程池拿到一个空闲线程,继续往下执行。
await Task.Delay(200);
//可理解等效成如下代码
await Task.Run(() =>
{
Thread.Sleep(200);
});
区别:
Thread.Sleep会阻塞当前线程,但是从线程池另取线程;Task.Delay不会阻塞当前线程,但是会新取一个线程
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的理解的更多相关文章
- async/await Task Timeout
async/await Task Timeout 在日常的电脑使用过程中,估计最难以忍受的就是软件界面"卡住""无响应",在我有限的开发生涯中一直都是在挑战 它 ...
- 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 ...
- async await Task
一.使用Task 引用命名空间 using System.Threading.Tasks; 1.工厂方式 Task.Factory.StartNew(() => {Console.WriteLi ...
- async,await,Task 的一些用法
async,await,Task 的一些用法 private void Form1_Load(object sender, EventArgs e) { Display(); } public asy ...
- C#扫盲篇(四):.NET Core 的异步编程-只讲干货(async,await,Task)
关于async,await,task的用法和解释这里就不要说明了,网上一查一大堆.至于为啥还要写这篇文章,主要是其他文章水分太多,不适合新手学习和理解.以下内容纯属个人理解,如果有误,请高手指正.本文 ...
- c#5.0(.net 4.5之后)的 Async+await+Task的异步机制的调试笔记
1.)无返回值的情况(异步也是基于线程). using System; using System.Collections.Generic; using System.Linq; using Syste ...
- Task.Delay() 和 Thread.Sleep() 区别
1.Thread.Sleep 是同步延迟,Task.Delay异步延迟. 2.Thread.Sleep 会阻塞线程,Task.Delay不会. 3.Thread.Sleep不能取消,Task.Dela ...
- task.delay 和 thread.sleep
1.Thread.Sleep 是同步延迟. Task.Delay异步延迟. 2.Thread.Sleep 会阻塞线程,Task.Delay不会. 3.Thread.Sleep不能取消,Task.Del ...
- async await Task 使用方法
使用概述 C#的使用过程中,除了以前的Thread.ThreadPool等用来开一个线程用来处理异步的内容.还可以使用新特性来处理异步.比以前的Thread和AutoResetEvent.delege ...
- async await task.Result 卡死
在如下代码中: public async Task<string> GetData() { return await DoWork(); } 在UI线程中调用 var data = Get ...
随机推荐
- RocketMQ实战—6.生产优化及运维方案
大纲 1.RocketMQ集群如何进行权限机制的控制 2.如何对RocketMQ集群进行消息堆积的追踪 3.如何处理RocketMQ的百万消息积压问题 4.针对RocketMQ集群崩溃的金融级高可用方 ...
- [业界方案] Yarn的业界解决方案和未来方向
[业界方案] Yarn的业界解决方案和未来方向 目录 [业界方案] Yarn的业界解决方案和未来方向 0x00 摘要 0x01 Yarn 1.1 参考文章 0x02 分析 2.1 综述 2.1.1 y ...
- Q:记录一次ssh毫无规律的断线
查找tailf /var/log/secure 网上一般是这三个原因 1.ssh服务端配置 vim /etc/ssh/sshd_config #客户端每隔多少秒向服务发送一个心跳数据,0代表不发送#C ...
- ORACLE11g数据中创建DB Link方法,用于跨oracle数据库查询数据
---查看该用户下已建立的DB link链接 SELECT * FROM DBA_DB_LINKS --创建语句 CREATE DATABASE LINK 连接名CONNECT TO 登录名 ID ...
- ORACLE SELECT INTO 赋值为空,抛出 NO DATA FOUND 异常
例子: DECLARE ORDER_NUM VARCHAR2(20); BEGIN SELECT S.ORDER_NUM INTO ORDER_NUM FROM SALES_ORDER S WHERE ...
- 小程序开发实战案例五 | 小程序如何嵌入H5页面
在接入小程序过程中会遇到需要将 H5 页面集成到小程序中情况,今天我们就来聊一聊怎么把 H5 页面塞到小程序中. 本篇文章将会从下面这几个方面来介绍: 小程序承载页面的前期准备 小程序如何承载 H5 ...
- .NET Core常用集合的几个坑
C#中的常见集合 注意,箭头线不代表继承关系,只代表功能上的加强,如有错误,欢迎指出. 泛型集合时间复杂度 集合类型 添加 删除 查找 访问(索引/键) 遍历 备注 List<T> O(1 ...
- Android开发之定时任务(AlarmManager、WorkManager)
Android 程序的定时任务主要有AlarmManager.WorkManager两种. 一.AlarmManager AlarmManager,又称闹钟,可以设置一次性任务,周期重复任务,定时重复 ...
- CentOS7安装部署ClickHouse(单机版&&集群部署)
1.1 什么是ClickHouse ClickHouse 是俄罗斯的Yandex于2016年开源的列式存储数据库(DBMS),主要用于在线分析处理查询(OLAP),能够使用SQL查询实时生成分析数据报 ...
- google浏览器删除token
测试登录时长,页面是否返回到首页 删除token