为了方便测试异步,先加个计时

计时相关(可以直接跳过该部分)

//开始计时

Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();

// 停止计时
stopwatch.Stop();

//输出计时毫秒数

stopwatch.ElapsedMilliseconds

阻塞延迟,下面两个分别表示阻塞三秒钟

同步阻塞:Thread.Sleep(TimeSpan.FromSeconds(3))

异步阻塞:Task.Delay(3000)

Thread.Sleep(TimeSpan.FromSeconds(3)):

所属命名空间:System.Threading.Thread

使用线程阻塞:调用该方法会导致当前线程被阻塞,即挂起当前线程的执行。

同步操作:Thread.Sleep 是同步的,意味着它会阻塞当前线程并占用系统资源。

精度较低:具体的暂停时间不一定非常准确,可能会稍微长一些。

Task.Delay(3000):

所属命名空间:System.Threading.Tasks.Task

使用任务异步操作:通过异步编程模型进行操作,异步等待指定的时间。

非阻塞操作:Task.Delay 是异步的,不会阻塞当前线程,而是让线程可以执行其他任务。

高精度定时器:提供更高的精度,具有更准确的延迟时间。

可以与 async/await 一起使用:await Task.Delay 可以与其他异步操作一起使用,提供更灵活和响应式的编程体验。

优缺点比较:

Thread.Sleep 的优点是简单直接,适用于简单的线程阻塞场景。然而,它会阻塞当前线程,并且在等待的过程中,当前线程将无法执行其他任务。

Task.Delay 的优点是允许异步等待一段时间,不会阻塞当前线程,可以与其他异步操作一起使用,提供更高的性能和响应能力。

此外,它还提供了更准确的定时精度。但是,如果不需要进行异步编程,或者代码基于旧版本.NET Framework,可能不适合使用。

介绍异步几个关键字

Task(任务):Task 是.NET Framework 和 .NET Core 中的一个类型,表示一个异步操作的单元。它可以被执行、等待和取消,并且可以返回结果。可以通过 Task.Run 方法创建一个 Task。

async/await 关键字:async/await 是 C# 5.0 引入的语言特性,用于简化异步编程。在方法定义前加上 async 关键字来表示这是一个异步方法,然后可以在方法体内使用 await 关键字来暂停方法的执行,等待一个异步操作完成,然后继续执行后续的代码。被 await 修饰的表达式必须是一个返回 Task 或 Task<T> 的异步方法调用。

await 关键字:await 关键字用于异步方法内部,表示等待一个异步操作完成。当遇到 await 关键字时,方法会立即返回给调用者,同时异步操作开始执行。等待的过程中,线程可以自由地执行其他任务。当异步操作完成后,该方法会恢复执行,并返回异步操作的结果。

Task.WhenAll 方法:Task.WhenAll 方法用于等待多个异步任务完成。它接受一个 Task 数组或可迭代对象作为参数,并返回一个新的 Task,该 Task 在所有传递的任务都完成后变为完成状态。你可以使用 await Task.WhenAll 来等待多个任务同时完成。

常见的属性

Task.Status 属性:表示任务的状态,有以下几种可能的值:

  1. Created:任务已创建但未开始执行。
  2. WaitingForActivation:任务在等待激活,即还未被调度器执行。
  3. WaitingToRun:任务已被调度器接受,并等待在可用线程上运行。
  4. Running:任务正在执行。
  5. WaitingForChildrenToComplete:父任务正在等待其所有子任务完成。
  6. RanToCompletion:任务成功完成。
  7. Faulted:任务发生了异常。
  8. Canceled:任务被取消。

Task.Result 属性:获取异步操作的返回结果。该属性只能在异步操作完成后使用,并且会阻塞当前线程直到异步操作完成。

下面上代码示例

async Task Main()
{
//计时
Stopwatch stopwatch = new Stopwatch();
// 开始计时
stopwatch.Start();
//执行顺序和添加顺序有关
await Task.WhenAll(A());//先执行A
await Task.WhenAll(B());//等待A执行完再执行B
List<Task> tasks = new List<Task>()
{ C(),D() };//等待B执行完后CD一起执行
//也可以逐个添加
//tasks.Add(A());
//tasks.Add(B());
//tasks.Add(C());
//tasks.Add(D()); // 等待CD方法全部完成
await Task.WhenAll(tasks);
// 停止计时
stopwatch.Stop();
Console.WriteLine($"计时结束!总共用时:{stopwatch.ElapsedMilliseconds} 毫秒");
} async Task A()
{
Console.WriteLine("阻塞3秒");
await Task.Delay(3000);
Console.WriteLine("3秒结束");
}
async Task B()
{
Console.WriteLine("阻塞2秒");
await Task.Delay(2000);
Console.WriteLine("2秒结束");
}
async Task C()
{
Console.WriteLine("阻塞5秒");
await Task.Delay(5000);
Console.WriteLine("5秒结束");
}
async Task D()
{
Console.WriteLine("阻塞10秒");
await Task.Delay(10000);
Console.WriteLine("10秒结束");
}

  

C# 异步执行操作的更多相关文章

  1. 深入理解 JS 引擎执行机制(同步执行、异步执行以及同步中的异步执行)

    首先明确两点: 1.JS 执行机制是单线程. 2.JS的Event loop是JS的执行机制,深入了解Event loop,就等于深入了解JS引擎的执行. 单线程执行带来什么问题? 在JS执行中都是单 ...

  2. [源码解析] PyTorch 分布式(16) --- 使用异步执行实现批处理 RPC

    [源码解析] PyTorch 分布式(16) --- 使用异步执行实现批处理 RPC 目录 [源码解析] PyTorch 分布式(16) --- 使用异步执行实现批处理 RPC 0x00 摘要 0x0 ...

  3. Java使用多线程异步执行批量更新操作

    import com.google.common.collect.Lists; import org.apache.commons.collections.CollectionUtils; impor ...

  4. [.NET] 利用 async & await 进行异步 IO 操作

    利用 async & await 进行异步 IO 操作 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6082673.html  序 上次,博主 ...

  5. android内部培训视频_第四节(1)_异步网络操作

    第四节(1):异步网络操作  一.结合asyncTask下载网络图片 1.定义下载类,继承自asyncTask,参数分别为:String(url地址),Integer(刻度,本例没有用到),BitMa ...

  6. Python开发程序:RPC异步执行命令(RabbitMQ双向通信)

    RPC异步执行命令 需求: 利用RibbitMQ进行数据交互 可以对多台服务器进行操作 执行命令后不等待命令的执行结果,而是直接让输入下一条命令,结果出来后自动打印 实现异步操作 不懂rpc的请移步h ...

  7. ajax同步、异步执行简单理解与证明

    此理解范例代码来自前几篇随笔! 首先我们来先了解下AJAX: Ajax:全称“Asynchronous Javascript and XML”(异步Javascript和XML),他是由Javascr ...

  8. node.js零基础详细教程(4):node.js事件机制、node异步IO操作

    第四章 建议学习时间3小时  课程共10章 学习方式:详细阅读,并手动实现相关代码 学习目标:此教程将教会大家 安装Node.搭建服务器.express.mysql.mongodb.编写后台业务逻辑. ...

  9. Go同步和异步执行多个任务封装

    同步执行类RunnerAsync 支持返回超时检测,系统中断检测 错误常量定义 //超时错误 var ErrTimeout = errors.New("received timeout&qu ...

  10. 获取node异步执行结果的方式

    拿数据库操作举例: var connection = mysql.createConnection(); connection.query(sql,function(err,rows){xxx} ); ...

随机推荐

  1. ET介绍—— 一切皆实体的设计

    一切皆实体 目前十分流行ECS设计,主要是守望先锋的成功,引爆了这种技术.守望先锋采用了状态帧这种网络技术,客户端会进行预测,预测不准需要进行回滚,由于组件式的设计,回滚可以只回滚某些组件即可.ECS ...

  2. 使用镜像加速 Rtools 下载与安装

    在 windows 使用 R,尤其是安装 R 包的时候,经常会遇到一些 Rtools 的问题,今天聊一下. Rtools 是什么 Rtools 作用很大,但我们一般不怎么会直接使用. Rtools p ...

  3. Linux服务器设置虚拟内存

    cd /usrsudo mkdir swapcd swapsudo dd if=/dev/zero of=/usr/swap/swapfile bs=1M count=4096du -sh /usr/ ...

  4. 使用poi-tl导出word文件的几个技巧

    1.前言   Poi-tl提供了基于word模板文件导出word文件的功能.文档地址:http://deepoove.com/poi-tl/.   用下来,总体感觉还是很方便的.但使用过程,有几个细节 ...

  5. Java并发(十一)----线程五种状态与六种状态

    1.五种状态 这是从 操作系统 层面来描述的 [初始状态]仅是在语言层面创建了线程对象,还未与操作系统线程关联 [可运行状态](就绪状态)指该线程已经被创建(与操作系统线程关联),可以由 CPU 调度 ...

  6. GO通道:无缓冲通道与缓冲通道

    转载请注明出处: 1.通道定义 在多个协程之间进行通信和管理,可以使用 Go 语言提供的通道(Channel)类型.通道是一种特殊的数据结构,可以在协程之间进行传递数据,从而实现协程之间的通信和同步. ...

  7. celery笔记六之worker介绍

    本文首发于公众号:Hunter后端 原文链接:celery笔记六之worker介绍 前面我们介绍过 celery 的理想的设计方式是几个 worker 处理特定的任务队列的数据,这样可以避免任务在队列 ...

  8. 开发中MongoDB遇到的各种问题

    目录 一.安装6版本以下 二.安装6版本及以上 三.安装6版本以下(解压版) 四.配置本地 Windows MongoGB 服务 五.navicat 连接远程mongodb数据库 六.ip不一致问题 ...

  9. linux150常用命令

    Linux最常用150个命令汇总 线上查询及帮助命令(2个) man 查看命令帮助,命令的词典,更复杂的还有info,但不常用. help 查看Linux内置命令的帮助,比如cd命令. 文件和目录操作 ...

  10. Federated Learning004

    联邦学习--笔记004 2023.03.13周一 快中期答辩了(3.20),最近甲流高发期 毕设期间,今天学习了联邦学习的一篇论文---Differentially Private Federated ...