C# 异步编程

异步编程是一种思路
异步相当于对线程池的封装
await相当于让另一个线程来干这个事	

  • 常见概念
    • 已经有多线程了,为何还要异步
      • 多线程与异步是不同的概念
      • 多线程与异步的适用场景不同
        • *多线程*
        • *异步*
    • 什么是异步任务(`Task`)
      • 包含了异步任务的各种状态的一个引用类型
      • 对于异步任务的抽象
      • 任务的结果
    • 异步方法(`async Task`)
          • *`async Task`*
    • 重要思想:不阻塞!
        • *常见阻塞情形*
    • 同步上下文
        • *`ConfigureAwait(false)`*
    • 一发即忘(Fire-and-forget)
  • 简单任务
    • 如何创建异步任务?
        • `Task.Run()`
        • `Task.Factory.StartNew()`
        • `new Task` + `Task.Start()`
        • 例子
    • 如何同时开启多个异步任务?
    • 任务如何取消?
    • 任务超时如何实现?
    • 在异步任务中汇报进度?
    • 如何在同步方法中调用异步方法?
  • 常见的误区
    • 异步一定是多线程?
    • 异步方法一定要写成async Task?
    • await一定会切换同步上下文?
    • 异步可以全面取代多线程?
    • Task.Result 一定会阻塞当前线程?
    • 开启的异步任务一定不会阻塞当前线程?
  • 同步机制
    • 传统方法
    • 轻量型
    • 并发集合
    • 第三方库



常见概念

已经有多线程了,为何还要异步

多线程与异步是不同的概念

异步并不意味着多线程,单线程同样可以异步
异步默认借助线程池
多线程经常阻塞,而异步要求不阻塞

多线程与异步的适用场景不同

多线程
适合CPU密集型操作
适合长期运行的任务
线程的创建与销毁开销较大
提供更底层的控制,操作线程、锁、信号量等
线程不易于传参及返回
线程的代码书写较为繁琐
异步
适合IO密集型操作
适合短暂的小任务
避免线程阻塞,提高系统响应能力

什么是异步任务(Task)

包含了异步任务的各种状态的一个引用类型

正在运行、完成、结果、报错等

public class TODO {public static event Func<object, string> foo;static void Main() {Task<string> task = new Task<string>((n) => {Thread.Sleep(1500);for (int i = 0; i < (int)n; i++) {Console.WriteLine("DONE {0}", i);}return "ok";}, 3);Console.WriteLine(task.Status);task.Start();Console.WriteLine(task.Status);Thread.Sleep(1000);Console.WriteLine(task.Status);Thread.Sleep(2000);Console.WriteLine(task.Status);Console.WriteLine(task.Result);}
}

输出结果

Created
WaitingToRun
Running
DONE 0
DONE 1
DONE 2
RanToCompletion
ok
另有ValueTask值类型版本

对于异步任务的抽象

开启异步任务后,当前线程并不会阻塞,而是可以去做其他事情
异步任务(默认)会借助线程池在其他线程上运行
获取结果后回到之前的状态

任务的结果

返回值为Task的方法表示异步任务没有返回值
返回值为Task<T>则表示有类型为T的返回值

异步方法(async Task)

将方法标记async 后,可以在方法中使用await关键字

await关键字会等待异步任务的结束,并获得结果

async + await 会将方法包装成状态机,await类似于检查点

MoveNext方法会被底层调用,从而切换状态
async Task
返回值依旧是Task类型,但是在其中可以使用await关键字
在其中写返回值可以直接写Task<T> 中的T类型,不用包装成Task<T>

async void

同样是状态机,但缺少记录状态的Task对象
无法聚合异常(Aggregate Exception),需要谨慎处理异常
几乎只用于对于事件的注册

*异步编程具有传染性(Contagious)
*
一处async,处处async
几乎所有自带方法都提供了异步的版本


重要思想:不阻塞!

await会暂时释放当前线程,使得该线程可以执行其他工作,而不必阻塞线程直到异步操作完成

不要在异步方法里用任何方式阻塞当前线程

常见阻塞情形

Task.Wait() & Task.Result

如果任务没有完成,则会阻塞当前线程,容易导致死锁Task.GetAwaiter().GetResult(),不会将Exception 包装为AggregateException

Task.Delay() vs. Thread.Sleep()

后者会阻塞当前的线程,这与异步编程的理念不符
前者是一个异步任务,会立刻释放当前的线程

IO等操作的同步方法

其他繁重且耗时的任务


同步上下文

一种管理和协调线程的机制,允许开发者将代码的执行切换到特定的线程

WinFormsWPF拥有同步上下文(UI 线程),而控制台程序默认没有

ConfigureAwait(false)
配置任务通过await方法结束后是否会到原来的线程,默认为true
一般只有UI线程会采用这种策略

TaskScheduler

控制Task的调度方式和运行线程

线程池线程Default
当前线程CurrentThread
单线程。上下文STAThread
长时间运行线程LongRunning

优先级、上下文、执行状态等


一发即忘(Fire-and-forget)

调用一一个异步方法,但是并不使用await或阻塞的方式去等待它的结束
无法观察任务的状态(是否完成、是否报错等)



简单任务

如何创建异步任务?

Task.Run()
Task.Factory.StartNew()
提供更多功能,比如TaskCreationOptions.L ongRunning
Task.Run相当于简化版
new Task + Task.Start()
很少有创建一个Task却没有让他立刻开始的
例子
public class TODO {static async Task Main() {Console.WriteLine(12);Console.WriteLine("ThreadId" + Environment.CurrentManagedThreadId.ToString());var task = await Task.Run(heavyJob);//Console.WriteLine(task);Console.WriteLine(123);}public static int heavyJob() {Console.WriteLine("ThreadId" + Environment.CurrentManagedThreadId.ToString());Thread.Sleep(10);return 1;}}

如何同时开启多个异步任务?

public class TODO {static async Task Main() {var inputs = Enumerable.Range(10, 10).ToArray();var tasks = new List<Task<int>>();Console.WriteLine(Environment.CurrentManagedThreadId);foreach (var input in inputs) {tasks.Add(foo(input));}await Task.WhenAll(tasks);var outputs = tasks.Select(x => x.Result).ToArray();foreach (var output in outputs) {Console.WriteLine(output);}}public static async Task<int> foo(int input) {await Task.Delay(5000);return input * 2;}
}

任务如何取消?

CancellationTokenSource + CancellationToken

public class TODO {static async Task Main() {var cts = new CancellationTokenSource();try {var task = Task.Delay(100000, cts.Token);Thread.Sleep(2000);cts.Cancel();//抛出异常await task;} catch (TaskCanceledException) {Console.WriteLine("ss");} finally {cts.Cancel();}}
}

OperationCanceledException & TaskCanceledException

推荐异步方法都带上CancellationToken这一传参

你自己写了异步方法却不支持传入这个————我可以不用,但不能没有

任务超时如何实现?


在异步任务中汇报进度?


如何在同步方法中调用异步方法?



常见的误区

异步一定是多线程?

异步编程不必需要多线程来实现

比如可以在单个线程上使用异步I/O 或事件驱动的编程模型(EAP)

单线程异步:自己定好计时器,到时间之前先去做别的事情
多线程异步:将任务交给不同的线程,并由自己来进行指挥调度


异步方法一定要写成async Task?

async关键字只是用来配合await 使用,从而将方法包装为状态机

本质上仍然是Task,只不过提供了语法糖,并且函数体中可以直接return Task的泛型类型

接口中无法声明async Task


await一定会切换同步上下文?

在使用await关键字调用并等待一个异步任务时,异步方法不一定会立刻来到新的线程上

如果await了一个已经完成的任务(包括Task.Delay(0)),会直接获得结果


异步可以全面取代多线程?

异步编程与多线程有一定关系,但两者并不是可以完全互相替代


Task.Result 一定会阻塞当前线程?

如果任务已经完成,那么Task.Result 可以直接得到结果


开启的异步任务一定不会阻塞当前线程?

await关键字不一定会立刻释放当前线程,所以如果调用的异步方法中存在阻塞(如Thread.Sleep(O))
那么依旧会阻塞当前上下文对应的线程



同步机制

传统方法

Monitor(lock)
Mutex
Semaphore
EventWaitHandle

轻量型

所有只有SemaphoreSlim不阻塞
ManualResetEventSlim

并发集合

第三方库

AsyncManulResetEvent

Miccrosoft.VisualStudio.Threading

AsyncLock

Nito.AsyncEx

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

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

相关文章

ArcGIS Pro 和 Python — 分析全球主要城市中心的土地覆盖变化

第一步——设置工作环境 1–0. 地理数据库 在下载任何数据之前,我将创建几个地理数据库,在其中保存和存储所有数据以及我将创建的后续图层。将为我要分析的五个城市中的每一个创建一个地理数据库,并将其命名为: “Phoenix.gdb” “Singapore.gdb” “Berlin.gdb” “B…

安卓悬浮窗权限检查

目录 悬浮窗权限代码检测悬浮窗功能 悬浮窗权限 请求了这个权限后&#xff0c;app的权限管理中会有「显示悬浮窗」的权限选项。后面会引导用户去开启这个权限。 <uses-permission android:name"android.permission.SYSTEM_ALERT_WINDOW" />代码检测悬浮窗功能…

Windows系统下将MySQL数据库表内的数据全量导入Elasticsearch

目录 下载安装Logstash 配置Logstash配置文件 运行配置文件 查看导入结果 使用Logstash将sql数据导入Elasticsearch 下载安装Logstash 官网地址 选择Windows系统&#xff0c;需下载与安装的Elasticsearch相同版本的&#xff0c;下载完成后解压安装包。 配置Logstash配…

贪吃蛇大作战【纯c语言】

如果有看到不懂的地方或者对c语言某些知识忘了的话&#xff0c;可以找我之前的文章哦&#xff01;&#xff01;&#xff01; 个人主页&#xff1a;小八哥向前冲~-CSDN博客 所属专栏&#xff1a;c语言_小八哥向前冲~的博客-CSDN博客 贪吃蛇游戏演示&#xff1a; 贪吃蛇游戏动画演…

第一阶段--Day2--信息安全法律法规、网络安全相关标准

目录 1. 针对信息安全的规定 2. 网络安全相关标准 1. 针对信息安全的规定 《中华人民共和国计算机信息系统安全保护条例》1994年2月18日颁布并实施 中华人民共和国计算机信息系统安全保护条例__增刊20111国务院公报_中国政府网 《中华人民共和国国际联网安全保护管理…

笔记:编写程序,分别采用面向对象和 pyplot 快捷函数的方式绘制正弦曲线 和余弦曲线。 提示:使用 sin()或 cos()函数生成正弦值或余弦值。

文章目录 前言一、面向对象和 pyplot 快捷函数的方式是什么&#xff1f;二、编写代码面向对象的方法&#xff1a;使用 pyplot 快捷函数的方法&#xff1a; 总结 前言 本文将探讨如何使用编程语言编写程序&#xff0c;通过两种不同的方法绘制正弦曲线和余弦曲线。我们将分别采用…

图像处理ASIC设计方法 笔记18 轮廓跟踪算法的硬件加速方案

目录 1排除伪孤立点(断裂链表)方法1 限制链表的长度方法2 增加判断条件排除断裂链表方法3 排除不必要跟踪的轮廓(推荐用这个方法)P129 轮廓跟踪算法的硬件加速方案 1排除伪孤立点(断裂链表) 如果图像中某区域存在相邻像素之间仅有对角连接的部位,则对包围该区域的像素…

SOLIDWORKS Electrical 3D--精准的三维布线

相信很多工程师在实际生产的时候都会遇到线材长度不准确的问题&#xff0c;从而导致线材浪费甚至整根线材报废的问题&#xff0c;这基本都是由于人工测量长度所导致的&#xff0c;因此本次和大家简单介绍一下SOLIDWORKS Electrical 3D布线的功能&#xff0c;Electrical 3D布线能…

伙伴匹配(后端)-- 用户登录

文章目录 登录逻辑设计登录业务代码实现用户登录态如何知道是哪个用户登录了&#xff1f;cookie与session 逻辑删除配置添加TableLogic注解 &#xff08;现在做单机登录&#xff09; 后面修改为redis单点登录 登录逻辑设计 接收参数&#xff1a;用户接账户&#xff0c;密码 请…

【数据标注】使用LabelImg标注YOLO格式的数据(案例演示)

文章目录 LabelImg介绍LabelImg安装LabelImg界面标注常用的快捷键标注前的一些设置案例演示检查YOLO标签中的标注信息是否正确参考文章 LabelImg介绍 LabelImg是目标检测数据标注工具&#xff0c;可以标注两种格式&#xff1a; VOC标签格式&#xff0c;标注的标签存储在xml文…

目标检测——蔬菜杂草数据集

引用 亲爱的读者们&#xff0c;您是否在寻找某个特定的数据集&#xff0c;用于研究或项目实践&#xff1f;欢迎您在评论区留言&#xff0c;或者通过公众号私信告诉我&#xff0c;您想要的数据集的类型主题。小编会竭尽全力为您寻找&#xff0c;并在找到后第一时间与您分享。 …

架构师系列- 消息中间件(12)-kafka基础

1、应用场景 1.1 kafka场景 Kafka最初是由LinkedIn公司采用Scala语言开发&#xff0c;基于ZooKeeper&#xff0c;现在已经捐献给了Apache基金会。目前Kafka已经定位为一个分布式流式处理平台&#xff0c;它以 高吞吐、可持久化、可水平扩展、支持流处理等多种特性而被广泛应用…

22年全国职业技能大赛——Web Proxy配置(web 代理)

前言&#xff1a;原文在我的博客网站中&#xff0c;持续更新数通、系统方面的知识&#xff0c;欢迎来访&#xff01; 系统服务&#xff08;22年国赛&#xff09;—— web Proxy服务&#xff08;web代理&#xff09;https://myweb.myskillstree.cn/114.html 目录 RouterSrv …

强复购、循环消费:排队复购模式助您在市场中脱颖而出

尊敬的各位读者&#xff0c;今天我很高兴向大家介绍一种新颖而又引人入胜的商业模式——排队复购模式。这个模式因其强大的复购属性和循环消费特性而备受瞩目&#xff0c;被誉为电商领域的新宠儿。 为何要介绍排队复购模式&#xff1f;因为它不仅操作简单、容易引起消费者的兴…

BUUCTF_[BSidesCF 2020]Had a bad day

[BSidesCF 2020]Had a bad day 1.一看题目直接尝试文件包含 2.直接报错&#xff0c;确实是存在文件包含漏洞 http://307b4461-36d6-443f-879a-68803a57f721.node5.buuoj.cn:81/index.php?categoryphp://filter/convert.base64-encode/resourceindex strpos() 函数查找字符串…

安卓玩机工具推荐----MTK芯片 简单制作线刷包 备份分区 备份基带 去除锁类 推荐工具操作解析

工具说明 在前面几期mtk芯片类玩机工具中解析过如何无官方固件从手机抽包 制作线刷包的步骤&#xff0c;类似的工具与操作有很多种。演示的只是本人片面的理解与一些步骤解析。mtk芯片机型抽包关键点在于..mt*****txt的分区地址段引导和 perloader临时分区引导。前面几期都是需…

【嵌入式Linux】STM32P1开发环境搭建

要进行嵌入式Linux开发&#xff0c;需要在Windows、Linux和嵌入式Linux3个系统之间来回跑&#xff0c;需要使用多个软件工具。经过了4小时的安装&#xff08;包括下载时间&#xff09;&#xff0c;我怕以后会忘记&#xff0c;本着互利互助的原则&#xff0c;我打算把这些步骤详…

java接口加密解密

这里写目录标题 controller加解密工具类加密&#xff08;本质是对ResponseBody加密&#xff09;解密&#xff08;本质是对RequestBody传参解密&#xff09;注解 controller Controller public class PathVariableController {GetMapping(value "/test")ResponseBod…

IDEA pom.xml依赖警告

IDEA中&#xff0c;有时 pom.xml 中会出现如下提示&#xff1a; IDEA 2022.1 升级了检测易受攻击的 Maven 和 Gradle 依赖项&#xff0c;并建议修正&#xff0c;通过插件 Package Checker 捆绑到 IDE 中。 这并不是引用错误&#xff0c;不用担心。如果实在强迫症不想看到这个提…

使用 FFmpeg 实现录屏和录音

FFmpeg 是一个非常强大的开源工具&#xff0c;可以用来处理音频和视频。可以实现录屏和录音&#xff0c;也可以进行简单的剪辑。 要使用 FFmpeg 进行录屏和录音&#xff0c;需要首先确保系统已经安装了 FFmpeg。在大多数 Linux 发行版中&#xff0c;可以通过包管理器&#xff0…