C#异步和多线程,Thread,Task和async/await关键字--12
目录
一.多线程和异步的区别
1.多线程
2.异步编程
多线程和异步的区别
二.Thread,Task和async/await关键字的区别
1.Thread
2.Task
3.async/await
三.Thread,Task和async/await关键字的详细对比
1.Thread和Task的详细对比
2.Task 与 async/await 的配合使用
3. async/await 的实际应用场景
4.关键区别总结
引言:在 C# 编程中,多线程和异步编程都是用于提高应用程序性能和响应性的技术,但它们的用途和实现方式有所不同
一.多线程和异步的区别
1.多线程
定义:
- 多线程是指在一个应用程序中同时运行多个线程,每个线程都独立执行代码.这些线程可能在多个处理器核心上并行运行,也可能在单个核心上通过时间片轮转方式运行
特性:
- 并行执行:多个线程可以并行执行任务,提高 CPU 利用率,适用于 CPU 密集型任务
- 共享内存空间:同一进程中的线程共享内存空间,可以方便地共享数据,但也需要处理同步问题
- 线程管理:需要开发者手动管理线程的创建,启动,同步和销毁
使用场景:
- 需要并行处理多个 CPU 密集型任务,以充分利用多核 CPU 的能力
- 需要在后台执行复杂计算,同时保持应用程序的响应性
2.异步编程
定义:
- 异步编程是一种编程范式,允许程序在等待长时间运行的操作(如 I/O、网络请求)完成时,不阻塞当前线程,从而提高应用程序的响应性和效率
特性:
- 非阻塞:异步操作不会阻塞线程,当前线程可以继续执行其他任务
- 事件驱动:通过回调,事件或任务的方式在操作完成时通知应用程序
- 适用于 I/O 密集型任务:特别是磁盘,网络等 I/O 操作,这些操作等待时间长,CPU 利用率低
使用场景:
- 防止界面卡顿:在 GUI 应用程序中,防止长时间的操作阻塞 UI 线程,保持界面响应性
- 服务器高并发:在服务器应用中,异步操作可以处理大量并发 I/O 请求,提高吞吐量
多线程和异步的区别
// 多线程示例
public void ThreadMethod()
{// 创建新线程Thread thread = new Thread(() =>{// 这段代码在新线程上运行DoSomeWork();});thread.Start();
}// 异步示例
public async Task AsyncMethod()
{// 不会创建新线程,而是在当前线程上异步执行await Task.Run(() =>{DoSomeWork();});
}
- 目的不同:多线程主要用于并行执行 CPU 密集型任务,异步编程主要用于非阻塞地执行 I/O 密集型任务
- 实现方式:多线程通过创建和管理线程实现并行,异步编程通过非阻塞的操作和回调机制实现,不一定需要多线程
- 资源利用:多线程可能会创建大量线程,占用系统资源;异步编程通常使用回调或任务,不需要额外的线程
// 多线程示例 - CPU密集型计算 public void ThreadExample() {Thread calculateThread = new Thread(() =>{// 复杂计算for (int i = 0; i < 1000000; i++){// 进行大量计算}});calculateThread.Start(); }// 异步示例 - I/O操作 public async Task AsyncExample() {// 读取文件string content = await File.ReadAllTextAsync("file.txt");// 发送网络请求var response = await httpClient.GetAsync("http://api.example.com"); }
二.Thread,Task和async/await关键字的区别
1.Thread
定义:
System.Threading.Thread
类表示 .NET 中的一个线程,可以用来手动创建,控制和管理线程
特性:
- 直接映射到操作系统线程:每个
Thread
实例对应一个操作系统线程 - 手动管理:需要手动创建,启动和销毁线程,以及处理线程同步问题
- 开销较大:创建和销毁线程的开销较大,过多的线程可能导致系统性能下降
使用场景:
- 需要对线程有精细的控制,比如设置线程优先级,堆栈大小,文化信息等
- 特定场景下需要手动管理线程的生命周期
示例代码:
// 创建第一个线程Thread thread1 = new Thread(() =>{for (int i = 0; i < 5; i++){Console.WriteLine($"线程1正在执行: {i}");Thread.Sleep(2000); // 暂停2秒}});// 创建第二个线程Thread thread2 = new Thread(new ThreadStart(Thread2Method));// 启动线程Console.WriteLine("开始执行线程...");thread1.Start();thread2.Start();// 等待线程结束//Join方法用于等待线程结束,即等待线程中的代码执行完毕。thread1.Join();thread2.Join();Console.WriteLine("所有线程执行完毕!");Console.ReadKey();
}// 第二个线程要执行的方法
static void Thread2Method()
{for (int i = 0; i < 5; i++){Console.WriteLine($"线程2正在执行: {i}");Thread.Sleep(800); // 暂停0.8秒}
}
2.Task
定义:
System.Threading.Tasks.Task
类代表一个异步操作,可以理解为更高级别的异步编程抽象
特性:
- 基于任务的异步模式(TAP):使用任务来表示异步操作,可以更方便地组合,链接和处理任务
- 线程池:
Task
默认会使用线程池中的线程,而不是创建新的线程,从而减少开销 - 支持结果值:
Task<TResult>
可以返回计算结果 - 与
async/await
协同工作:Task
可以与async/await
关键字配合使用,简化异步编程
使用场景:
- 执行异步操作,无需手动管理线程。
- 需要组合多个异步操作,或处理异步操作的结果。
static async Task Main(string[] args){Console.WriteLine("开始任务示例...");// 1. 基本的异步任务await SimpleTaskAsync();// 2. 带返回值的任务int result = await CalculateAsync();Console.WriteLine($"计算结果: {result}");// 3. 并行任务await ParallelTasksAsync();// 4. 任务超时处理await TaskWithTimeoutAsync();Console.WriteLine("所有任务完成!");Console.ReadKey();}// 基本异步任务static async Task SimpleTaskAsync(){Console.WriteLine("开始简单任务");await Task.Delay(1000); // 模拟耗时操作Console.WriteLine("简单任务完成");}// 带返回值的异步任务static async Task<int> CalculateAsync(){Console.WriteLine("开始计算");await Task.Delay(2000); // 模拟复杂计算return 42;}// 并行任务示例static async Task ParallelTasksAsync(){Console.WriteLine("开始并行任务");var task1 = Task.Run(async () =>{await Task.Delay(1000);Console.WriteLine("任务1完成");});var task2 = Task.Run(async () =>{await Task.Delay(2000);Console.WriteLine("任务2完成");});// 等待所有任务完成await Task.WhenAll(task1, task2);Console.WriteLine("所有并行任务完成");}// 带超时的任务static async Task TaskWithTimeoutAsync(){Console.WriteLine("开始超时任务");try{using var cts = new CancellationTokenSource(TimeSpan.FromSeconds(2));await Task.Delay(3000, cts.Token); // 这个任务会超时Console.WriteLine("此行不会执行");}catch (TaskCanceledException){Console.WriteLine("任务已超时");}}
3.async/await
定义:async/
await
是 C# 中用于简化异步编程的关键字
特性:
async
方法:使用async
修饰的方法表示其内部可能包含异步操作,可以使用await
关键字await
关键字:用于等待一个异步任务完成,而不会阻塞当前线程- 编译器支持:编译器会将
async
方法转换为状态机,处理异步操作的调度
使用场景:
- 希望以同步的方式编写异步代码,提高代码的可读性和维护性。
- 在 GUI 或服务器应用中,防止长时间的操作阻塞线程。
三.Thread,Task和async/await关键字的详细对比
1.Thread和Task的详细对比
// Thread 示例
Thread thread = new Thread(() =>
{Console.WriteLine("使用 Thread 执行工作");Thread.Sleep(1000);
});
thread.Start();// Task 示例
Task task = Task.Run(() =>
{Console.WriteLine("使用 Task 执行工作");Thread.Sleep(1000);
});
主要区别:
Thread:
- 直接映射到操作系统线程
- 资源开销大
- 无法直接返回结果
- 不易于管理和组合
Task:
- 使用线程池
- 可以返回结果
- 支持取消、延续、异常处理
- 易于组合和管理
2.Task 与 async/await 的配合使用
// Task 单独使用
public Task<int> GetDataAsync()
{return Task.Run(() => {// 执行一些耗时操作Thread.Sleep(1000);return 42;});
}// 使用 async/await
public async Task<int> GetDataAsyncWithAwait()
{Console.WriteLine("开始");await Task.Delay(1000); // 异步等待Console.WriteLine("结束");return 42;
}
3. async/await 的实际应用场景
// 文件操作示例
public async Task SaveFileAsync(string content)
{Console.WriteLine("开始保存文件");await File.WriteAllTextAsync("test.txt", content);Console.WriteLine("文件保存完成");
}// 多个异步操作组合
public async Task ProcessDataAsync()
{try{// 并行执行多个异步操作var task1 = Task.Delay(1000);var task2 = Task.Delay(2000);await Task.WhenAll(task1, task2);// 串行执行异步操作var result1 = await GetDataAsync();var result2 = await ProcessResultAsync(result1);}catch (Exception ex){Console.WriteLine($"错误: {ex.Message}");}
}
4.关键区别总结
1)执行模型:
- Thread: 一个线程执行一个任务
- Task: 可以使用线程池,更灵活
- async/await: 不创建新线程,而是管理异步操作
2)资源使用:
- Thread: 每个线程占用约1MB内存
- Task: 更轻量级,共享线程池
- async/await: 几乎没有额外开销
3)使用场景:
- Thread: 需要直接控制线程时
- Task: 执行后台操作,需要返回结果时
- async/await: IO操作,网络请求等不需要CPU计算的操作
4)代码可维护性:
- Thread: 较难管理和维护
- Task: 提供更好的控制和组合
- async/await: 提供最清晰的代码结构
在实际的开发中优先使用 async/await 处理异步操作,需要并行计算时使用 Task,只在特殊情况下使用 Thread
相关文章:
C#异步和多线程,Thread,Task和async/await关键字--12
目录 一.多线程和异步的区别 1.多线程 2.异步编程 多线程和异步的区别 二.Thread,Task和async/await关键字的区别 1.Thread 2.Task 3.async/await 三.Thread,Task和async/await关键字的详细对比 1.Thread和Task的详细对比 2.Task 与 async/await 的配合使用 3. asy…...
使用分割 Mask 和 K-means 聚类获取天空的颜色
引言 在计算机视觉领域,获取天空的颜色是一个常见任务,广泛应用于天气分析、环境感知和图像增强等场景。本篇博客将介绍如何通过已知的天空区域 Mask 提取天空像素,并使用 K-means 聚类分析天空颜色,最终根据颜色占比查表得到主导…...
145.《redis原生超详细使用》
文章目录 什么是redisredis 安装启动redis数据类型redis key操作key 的增key 的查key 的改key 的删key 是否存在key 查看所有key 「设置」过期时间key 「查看」过期时间key 「移除」过期时间key 「查看」数据类型key 「匹配」符合条件的keykey 「移动」到其他数据库 redis数据类…...

Pytorch基础教程:从零实现手写数字分类
文章目录 1.Pytorch简介2.理解tensor2.1 一维矩阵2.2 二维矩阵2.3 三维矩阵 3.创建tensor3.1 你可以直接从一个Python列表或NumPy数组创建一个tensor:3.2 创建特定形状的tensor3.3 创建三维tensor3.4 使用随机数填充tensor3.5 指定tensor的数据类型 4.tensor基本运算…...

【SH】Xiaomi9刷Windows10系统研发记录 、手机刷Windows系统教程、小米9重装win10系统
文章目录 参考资料云盘资料软硬件环境手机解锁刷机驱动绑定账号和设备解锁手机 Mindows工具箱安装工具箱和修复下载下载安卓和woa资源包第三方Recovery 一键安装Windows准备工作创建分区安装系统 效果展示Windows和Android一键互换Win切换安卓安卓切换Win 删除分区 参考资料 解…...

excel仅复制可见单元格,仅复制筛选后内容
背景 我们经常需要将内容分给不同的人,做完后需要合并 遇到情况如下 那是因为直接选择了整列,当然不可以了。 下面提供几种方法,应该都可以 直接选中要复制区域然后复制,不要选中最上面的列alt;选中可见单元格正常复制ÿ…...
HBASE学习(一)
1.HBASE基础架构, 1.1 参考: HBase集群架构与读写优化:理解核心机制与性能提升-CSDN博客 1.2问题: 1.FLUSH对hbase的影响 2. HLog和memstore的区别 hlog中存储的是操作记录,比如写、删除。而memstor中存储的是写入…...

element select 绑定一个对象{}
背景: select组件的使用,适用广泛的基础单选 v-model 的值为当前被选中的 el-option 的 value 属性值。但是我们这里想绑定一个对象,一个el-option对应的对象。 <el-select v-model"state.form.modelA" …...

Sprint Boot教程之五十八:动态启动/停止 Kafka 监听器
Spring Boot – 动态启动/停止 Kafka 监听器 当 Spring Boot 应用程序启动时,Kafka Listener 的默认行为是开始监听某个主题。但是,有些情况下我们不想在应用程序启动后立即启动它。 要动态启动或停止 Kafka Listener,我们需要三种主要方法…...
C:JSON-C简介
介绍 JSON-C是一个用于处理JSON格式数据的C语言库,提供了一系列操作JSON数据的函数。 一、json参数类型 typedef enum json_type { json_type_null, json_type_boolean, json_type_double, json_type_int, json_type_object, json_type_ar…...

业务幂等性技术架构体系之消息幂等深入剖析
在系统中当使用消息队列时,无论做哪种技术选型,有很多问题是无论如何也不能忽视的,如:消息必达、消息幂等等。本文以典型的RabbitMQ为例,讲解如何保证消息幂等的可实施解决方案,其他MQ选型均可参考。 一、…...

【Go】Go Gin框架初识(一)
1. 什么是Gin框架 Gin框架:是一个由 Golang 语言开发的 web 框架,能够极大提高开发 web 应用的效率! 1.1 什么是web框架 web框架体系图(前后端不分离)如下图所示: 从上图中我们可以发现一个Web框架最重要…...
2024年合肥市科普日小学组市赛第一题题解
9304:数字加密(encrypt)(1) 【问题描述】 在信息科技课堂上,小肥正在思考“数字加密”实验项目。项目需要加密n个正整数,对每一个正整数x加密的规则是,将x的每一位数字都替换为x的最大数字。例如࿰…...

【MySQL实战】mysql_exporter+Prometheus+Grafana
要在Prometheus和Grafana中监控MySQL数据库,如下图: 可以使用mysql_exporter。 以下是一些步骤来设置和配置这个监控环境: 1. 安装和配置Prometheus: - 下载和安装Prometheus。 - 在prometheus.yml中配置MySQL通过添加以下内…...
Wireshark 使用教程:网络分析从入门到精通
一、引言 在网络技术的广阔领域中,网络协议分析是一项至关重要的技能。Wireshark 作为一款开源且功能强大的网络协议分析工具,被广泛应用于网络故障排查、网络安全检测以及网络协议研究等诸多方面。本文将深入且详细地介绍 Wireshark 的使用方法&#x…...

如何在前端给视频进行去除绿幕并替换背景?-----Vue3!!
最近在做这个这项目奇店桶装水小程序V1.3.9安装包骑手端V2.0.1小程序前端 最近,我在进行前端开发时,遇到了一个难题“如何给前端的视频进行去除绿幕并替换背景”。这是一个“数字人项目”所需,我一直在冥思苦想。终于有了一个解决方法…...
使用中间件自动化部署java应用
为了实现你在 IntelliJ IDEA 中打包项目并通过工具推送到两个 Docker 服务器(172.168.0.1 和 172.168.0.12),并在推送后自动或手动重启容器,我们可以按照以下步骤进行操作: 在 IntelliJ IDEA 中配置 Maven 或 Gradle 打…...
pytorch张量分块投影示例代码
张量的投影操作 背景 张量投影 是深度学习中常见的操作,将输入张量通过线性变换映射到另一个空间。例如: Y=W⋅X+b 其中: X: 输入张量(形状可能为 (B,M,K),即批量维度、序列维度、特征维度)。W: 权重矩阵((K,N),将 K 维投影到 N 维)。b: 偏置向量(可选,(N,))。Y:…...

Visual Studio 同一解决方案 同时运行 多个项目
方案一 方案二...
VMware中Ubuntu如何连接网络?安排!
一、设置NAT模式 1、关闭Ubuntu虚拟机: 确保Ubuntu已经完全关机,而不是挂起或休眠状态。 2、编辑虚拟网络设置: 在VMware主界面点击“编辑”菜单,选择“虚拟网络编辑器”。 如果需要,选择VMnet8 (NAT模式)并点击“更改…...
Vue记事本应用实现教程
文章目录 1. 项目介绍2. 开发环境准备3. 设计应用界面4. 创建Vue实例和数据模型5. 实现记事本功能5.1 添加新记事项5.2 删除记事项5.3 清空所有记事 6. 添加样式7. 功能扩展:显示创建时间8. 功能扩展:记事项搜索9. 完整代码10. Vue知识点解析10.1 数据绑…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...

【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...

前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...

多种风格导航菜单 HTML 实现(附源码)
下面我将为您展示 6 种不同风格的导航菜单实现,每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
以下是一个完整的 Angular 微前端示例,其中使用的是 Module Federation 和 npx-build-plus 实现了主应用(Shell)与子应用(Remote)的集成。 🛠️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...

算法:模拟
1.替换所有的问号 1576. 替换所有的问号 - 力扣(LeetCode) 遍历字符串:通过外层循环逐一检查每个字符。遇到 ? 时处理: 内层循环遍历小写字母(a 到 z)。对每个字母检查是否满足: 与…...