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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

fastapi 使用

参考&#xff1a; https://fastapi.tiangolo.com/zh/tutorial/first-steps/https://fastapi.tiangolo.com/zh/tutorial/first-steps/ FastAPI 用于基于标准 Python 类型提示使用 Python 构建 API&#xff0c;使用 ASGI 的标准来构建 Python Web 框架和服务器。所有简单理解&a…

2024年度漏洞态势分析报告,需要访问自取即可!(PDF版本)

2024年度漏洞态势分析报告&#xff0c;需要访问自取即可!(PDF版本),大家有什么好的也可以发一下看看

泛目录和泛站有什么差别

啥是 SEO 泛目录&#xff1f; 咱先来说说 SEO 泛目录是啥。想象一下&#xff0c;你有一个巨大的图书馆&#xff0c;里面的书架上摆满了各种各样的书&#xff0c;每一本书都代表着一个网页。而 SEO 泛目录呢&#xff0c;就像是一个超级图书管理员&#xff0c;它的任务就是把这些…

k8s基础(6)—Kubernetes-存储

Kubernetes-存储概述 k8s的持久券简介 Kubernetes的持久卷&#xff08;PersistentVolume, PV&#xff09;和持久卷声明&#xff08;PersistentVolumeClaim, PVC&#xff09;为用户在Kubernetes中使用卷提供了抽象。PV是集群中的一块存储&#xff0c;PVC是对这部分存储的请求。…

深度学习-卷积神经网络反向传播梯度公式推导

这篇文章非常棒&#xff0c;单样本单通道的反向传播梯度公式推导我都理解了。为了防止找不到原网页&#xff0c;所以特复制于此 参考&#xff1a; https://zhuanlan.zhihu.com/p/640697443

论文笔记(四十七)Diffusion policy: Visuomotor policy learning via action diffusion(下)

Diffusion policy: Visuomotor policy learning via action diffusion&#xff08;下&#xff09; 文章概括5. 评估5.1 模拟环境和数据集5.2 评估方法论5.3 关键发现5.4 消融研究 6 真实世界评估6.1 真实世界Push-T任务6.2 杯子翻转任务6.3 酱汁倒入和涂抹任务 7. 实际双臂任务…

C#学习笔记 --- 简单应用

1.operator 运算符重载&#xff1a;使自定义类可以当做操作数一样进行使用。规则自己定。 2.partial 分部类&#xff1a; 同名方法写在不同位置&#xff0c;可以当成一个类使用。 3.索引器&#xff1a;使自定义类可以像数组一样通过索引值 访问到对应的数据。 4.params 数…

汽车基础软件AutoSAR自学攻略(四)-AutoSAR CP分层架构(3) (万字长文-配21张彩图)

汽车基础软件AutoSAR自学攻略(四)-AutoSAR CP分层架构(3) (万字长文-配21张彩图) 前面的两篇博文简述了AutoSAR CP分层架构的概念&#xff0c;下面我们来具体到每一层的具体内容进行讲解&#xff0c;每一层的每一个功能块力求用一个总览图&#xff0c;外加一个例子的图给大家进…

【2024年华为OD机试】 (CD卷,100分)- 最大N个数与最小N个数的和(Java JS PythonC/C++)

一、问题描述 题目描述 给定一个数组&#xff0c;编写一个函数来计算它的最大N个数与最小N个数的和。你需要对数组进行去重。 说明&#xff1a; 数组中数字范围 [0, 1000]最大N个数与最小N个数不能有重叠&#xff0c;如有重叠&#xff0c;输入非法返回 -1输入非法返回 -1 …

WINFORM - DevExpress -> DevExpress总结[安装、案例]

安装devexpress软件 路径尽量不换&#xff0c;后面破解不容易出问题 vs工具箱添加控件例如: ①使用控制台进入DevExpress安装目录: cd C:\Program Files (x86)\DevExpress 20.1\Components\Tools ②添加DevExpress控件&#xff1a; ToolboxCreator.exe/ini:toolboxcreator…

cursor+deepseek构建自己的AI编程助手

文章目录 准备工作在Cursor中添加deepseek 准备工作 下载安装Cursor &#xff08;默认安装在C盘&#xff09; 注册deepseek获取API key 在Cursor中添加deepseek 1、打开cursor&#xff0c;选择设置 选择Model&#xff0c;添加deepseek-chat 注意这里去掉其他的勾选项&…

《零基础Go语言算法实战》【题目 2-7】defer 关键字特性

《零基础Go语言算法实战》 【题目 2-7】defer 关键字特性 下面代码的输出是什么&#xff1f;请说明原因。 package main import ( "fmt" ) func main() { deferFunc() func deferFunc() { defer func() { fmt.Println("value1") }() defer func() {…

如何规模化实现完全自动驾驶?Mobileye提出解题“新”思路

在CES 2025上&#xff0c;Mobileye展示了端到端自动驾驶系统Mobileye Drive™&#xff0c;通过高度集成的传感器、算法和计算平台&#xff0c;可以实现自动驾驶功能的全覆盖。 Mobileye创始人兼首席执行官Amnon Shashua教授 期间&#xff0c;Mobileye创始人兼首席执行官Amnon …

腾讯云AI代码助手编程挑战赛-智能聊天助手

作品简介 本作品开发于腾讯云 AI 代码助手编程挑战赛&#xff0c;旨在体验腾讯云 AI 代码助手在项目开发中的助力。通过这一开发过程&#xff0c;体验到了 AI 辅助编程的高效性。 技术架构 前端: 使用 VUE3、TypeScript、TDesign 和 ElementUI 实现。 后端: 基于 Python 开发…

超大规模分类(三):KNN softmax

传统的分类损失计算输入数据和每个类别中心的距离&#xff0c;来优化模型的训练。KNN softmax通过选择和输入数据最相关的top-K个类别&#xff0c;仅计算输入数据和top-K个类别中心的距离&#xff0c;以减小计算量。 KNN softmax首次诞生于达摩院机器智能技术实验室发表的SIGKD…

MySQL素材怎么导入Navicat???

不管用什么方法都要先关掉MySQL服务&#xff0c;并且提前备份数据&#xff01; 1.有sql文件时候。 打开navicat&#xff0c;运行sql文件 然后点击后面三个点&#xff0c;选中要运行的sql文件&#xff0c;开始。 鼠标右键刷新一下&#xff0c;就能看到sql文件中的表了 2.没有s…

程序员独立开发竞品分析:确定网站使用什么建站系统

要确定一个网站使用的建站系统&#xff0c;可以通过以下几种方法尝试分析&#xff1a; 查看页面源代码&#xff1a; 打开网站&#xff0c;右键点击页面并选择“查看页面源代码”。在代码中查找一些常见的建站系统标志&#xff0c;例如&#xff1a; WordPress 的迹象&#xff1a…

Linux(Centos7)安装Mysql/Redis/MinIO

安装Mysql 安装Redis 搜索Redis最先版本所在的在线安装yum库 查看以上两个组件是否是开机自启 安装MinIO 开源的对象存储服务&#xff0c;存储非结构化数据&#xff0c;兼容亚马逊S3协议。 minio --help #查询命令帮助minio --server --help #查询--server帮助minio serve…

【DB-GPT】开启数据库交互新篇章的技术探索与实践

一、引言&#xff1a;AI原生数据应用开发的挑战与机遇 在数字化转型的浪潮中&#xff0c;企业对于智能化应用的需求日益增长。然而&#xff0c;传统的数据应用开发方式面临着诸多挑战&#xff0c;如技术栈复杂、开发周期长、成本高昂、难以维护等。这些问题限制了智能化应用的…

解决aerich init -t xx 报错ModuleNotFoundError: No module named ‘tomli_w‘

今天在学习fastapi的时候&#xff0c;发现一款数据库迁移工具&#xff0c;通过这个工具可以根据模型类来对数据库做出改变。 随跟着学: 在执行 aerich init -t settings.TORTOISE_ORM的时候&#xff0c; 彼其娘之。。 报了一些错误&#xff1a; Traceback (most recent ca…