当前位置: 首页 > news >正文

C#中多线程编程中的同步、异步、串行、并行及并发及死锁

在C#中,多线程编程是一个强大的功能,它允许程序同时执行多个任务。然而,这也带来了复杂性,特别是在处理同步、异步、串行、并行、并发以及死锁等问题时。下面我将详细解释这些概念,并给出一些C#中的示例和注意事项。

目录

1. 同步(Synchronous)

2. 异步(Asynchronous)

3. 串行(Serial)

4. 并行(Parallel)

5. 并发(Concurrency)

6. 死锁(Deadlock)

结论


1. 同步(Synchronous)

在同步编程中,任务的执行顺序是线性的,即一个任务完成后,下一个任务才会开始。C#中的同步方法会阻塞调用线程,直到方法执行完成。

示例

void SynchronousMethod()
{// 执行一些操作Thread.Sleep(1000); // 模拟耗时操作Console.WriteLine("SynchronousMethod 完成");
}

2. 异步(Asynchronous)

异步编程允许任务在后台执行,而不会阻塞调用线程。在C#中,asyncawait关键字用于实现异步编程。

示例

async Task AsynchronousMethod()
{// 等待异步操作完成await Task.Delay(1000); // 模拟耗时操作Console.WriteLine("AsynchronousMethod 完成");
}

3. 串行(Serial)

串行执行意味着任务按照顺序一个接一个地执行,这在单线程程序中是自然的。

// 同步串行
void Task1()
{Console.WriteLine("Task 1 is running");
}void Task2()
{Console.WriteLine("Task 2 is running");
}Task1();
Task2();// 异步串行
async Task Task1Async()
{Console.WriteLine("Task 1 is running");await Task.Delay(1000); // 模拟异步操作
}async Task Task2Async()
{Console.WriteLine("Task 2 is running");await Task.Delay(1000); // 模拟异步操作
}async Task MainAsync()
{await Task1Async();await Task2Async();
}MainAsync().GetAwaiter().GetResult(); // 运行异步方法

4. 并行(Parallel)

并行执行允许多个任务同时执行,通常利用多核处理器的能力。在C#中,Parallel.ForParallel.ForEachParallel.Invoke等API用于并行执行。

示例

using System.Threading.Tasks;void Task1()
{Console.WriteLine("Task 1 is running");
}void Task2()
{Console.WriteLine("Task 2 is running");
}Parallel.Invoke(() => Task1(), () => Task2());
Parallel.For(0, 10, i =>
{Console.WriteLine($"并行任务 {i} 开始");Thread.Sleep(100); // 模拟耗时操作Console.WriteLine($"并行任务 {i} 完成");
});

5. 并发(Concurrency)

并发是并行和串行在更广泛意义上的结合。它指的是多个任务同时或几乎同时执行,但不一定在物理上并行(可能由于时间片轮转而在单核处理器上模拟并行)。

using System.Threading;void Task1()
{Console.WriteLine("Task 1 is running");
}void Task2()
{Console.WriteLine("Task 2 is running");
}Thread t1 = new Thread(new ThreadStart(Task1));
Thread t2 = new Thread(new ThreadStart(Task2));t1.Start();
t2.Start();t1.Join();
t2.Join();

6. 死锁(Deadlock)

死锁是多线程编程中常见的问题,当两个或多个线程相互等待对方释放资源时,就会发生死锁。这会导致这些线程永远无法继续执行。

死锁发生的四个必要条件:

互斥条件:指进程对所分配到的资源进行排它性使用,即在一段时间内某资源只由一个进程占用。如果此时还有其它进程请求资源,则请求者只能等待,直至占有资源的进程用毕释放。

请求和保持条件:指进程已经保持至少一个资源,但又提出了新的资源请求,而该资源已被其它进程占有,此时请求进程阻塞,但又对自己已获得的其它资源保持不放。

不剥夺条件:指进程已获得的资源,在未使用完之前,不能被剥夺,只能在使用完时由自己释放。
环路等待条件:指在发生死锁时,必然存在一个进程——资源的环形链,即进程集合{P0,P1,P2,···,Pn}中的P0正在等待一个P1占用的资源,P1正在等待P2占用的资源,……,Pn正在等待已被P0占用的资源

避免死锁的策略

  • 确保所有线程以相同的顺序获取资源。

  • 使用锁超时机制。

  • 避免嵌套锁。

  • 使用更高级的并发控制机制,如信号量、事件或Concurrent集合。

示例(可能导致死锁):

object lock1 = new object();
object lock2 = new object();void Thread1Method()
{lock (lock1){Thread.Sleep(100); // 模拟耗时操作lock (lock2){// 执行一些操作}}
}void Thread2Method()
{lock (lock2){Thread.Sleep(100); // 模拟耗时操作lock (lock1){// 执行一些操作}}
}

在这个例子中,如果Thread1MethodThread2Method几乎同时执行,并且都尝试先锁定lock1lock2(但顺序相反),那么它们可能会相互等待对方释放锁,从而导致死锁。

结论

在C#中进行多线程编程时,需要仔细考虑同步、异步、串行、并行和并发的问题,以及如何避免死锁等并发问题。合理使用asyncawaitParallel类、锁(如lock语句)以及Concurrent集合等,可以帮助你编写高效且稳定的多线程程序。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • <数据集>航拍车辆识别数据集<目标检测>
  • RocketMQ Dashboard安装
  • 逻辑推理复杂推理能力评估之baseline详解(Datawhale AI 夏令营)
  • C++ 微积分 - 求导 - 解析法(符号计算)
  • Laravel为什么会成为最优雅的PHP框架?
  • B - 02-计算球的体积 51Nod - 3266
  • Python可视化开发全面教程
  • 使用 1panel面板 部署 php网站
  • 电路元件基本知识详解
  • 基础实验回顾
  • 【密码学】椭圆曲线密码体制(ECC)
  • 基于深度学习的联邦学习
  • 【Python】任推邦近30日推广数据采集+推送
  • Open3D 计算点云的归一化协方差矩阵
  • JAVA中的重载
  • [数据结构]链表的实现在PHP中
  • E-HPC支持多队列管理和自动伸缩
  • GraphQL学习过程应该是这样的
  • HomeBrew常规使用教程
  • Laravel5.4 Queues队列学习
  • Laravel核心解读--Facades
  • MYSQL 的 IF 函数
  • Python实现BT种子转化为磁力链接【实战】
  • spring boot下thymeleaf全局静态变量配置
  • spring cloud gateway 源码解析(4)跨域问题处理
  • SQLServer之创建数据库快照
  • Transformer-XL: Unleashing the Potential of Attention Models
  • 从地狱到天堂,Node 回调向 async/await 转变
  • 当SetTimeout遇到了字符串
  • 动手做个聊天室,前端工程师百无聊赖的人生
  • 浮现式设计
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 简单实现一个textarea自适应高度
  • 罗辑思维在全链路压测方面的实践和工作笔记
  • 七牛云假注销小指南
  • 人脸识别最新开发经验demo
  • 如何设计一个比特币钱包服务
  • 探索 JS 中的模块化
  • 跳前端坑前,先看看这个!!
  • 协程
  • 原生JS动态加载JS、CSS文件及代码脚本
  • 在GitHub多个账号上使用不同的SSH的配置方法
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • 从如何停掉 Promise 链说起
  • ​ ​Redis(五)主从复制:主从模式介绍、配置、拓扑(一主一从结构、一主多从结构、树形主从结构)、原理(复制过程、​​​​​​​数据同步psync)、总结
  • #我与Java虚拟机的故事#连载11: JVM学习之路
  • (LeetCode) T14. Longest Common Prefix
  • (读书笔记)Javascript高级程序设计---ECMAScript基础
  • (个人笔记质量不佳)SQL 左连接、右连接、内连接的区别
  • (九)One-Wire总线-DS18B20
  • (论文阅读笔记)Network planning with deep reinforcement learning
  • (三十五)大数据实战——Superset可视化平台搭建
  • (十五)使用Nexus创建Maven私服
  • (四)Controller接口控制器详解(三)
  • (算法)Game