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

多个线程多个锁:如何确保线程安全和避免竞争条件

目录

前言

一、确定需要多个锁的场景

1.独立资源保护

2.部分依赖资源

二、避免死锁

三、锁粒度与并发性能

1. 粗粒度锁定

2.细粒度锁定

四、设计策略:减少资源依赖

1.资源分离

2.无锁设计

3.锁合并

五、Demo讲解

总结:


前言

        当多个线程需要操作共享资源时,为了确保数据的一致性和避免竞争条件,通常会使用多个锁来进行同步。这种情况下,如何正确使用多个锁成为一个复杂而关键的问题。下面是一篇十分详细的博客,介绍多线程多锁场景下的最佳实践和注意事项。

一、确定需要多个锁的场景

1.独立资源保护

  • 定义:当不同的资源(例如文件、数据库连接等)由不同的锁保护时。
  • 示例:一个线程需要读取文件A并写入文件B,而另一个线程读取文件B并写入文件A,这两个操作可以分别使用不同的锁。

2.部分依赖资源

  • 定义:多个资源之间存在某种程度的依赖关系,但操作它们的线程可能不会同时访问所有资源。
  • 示例:两个线程分别操作两个互相有数据交换的队列,可分别对两个队列加锁,但在交换数据时需要特别小心处理锁的顺序。

二、避免死锁

死锁是多线程编程中常见的问题,特别是在使用多个锁的情况下更容易发生。要避免死锁,可以采取以下策略:

  • 按顺序获取锁:对多个资源使用相同的顺序获取锁,以避免循环等待。
  • 设置超时时间:在获取锁的过程中设置超时时间,一段时间后未能获取到锁就放弃或重试。
  • 使用高级同步工具:比如信号量(Semaphores)或条件变量(Condition Variables),它们提供了更灵活的同步机制,有助于避免死锁。

三、锁粒度与并发性能

1. 粗粒度锁定

  • 优点:实现简单,易于理解和维护。
  • 缺点:可能导致大量线程等待,从而降低并发性能。
  • 示例:一个单一的大锁保护整个资源集合。

2.细粒度锁定

  • 优点:提高并发性能,因为锁的范围缩小,减少了线程等待的概率。
  • 缺点:实现复杂,需要更精细的设计和管理。
  • 示例:为每个独立资源(或资源的部分)使用单独的小锁。

四、设计策略:减少资源依赖

1.资源分离

  • 定义:尽量将共享资源划分为独立的部分,使得每个部分只需一个锁。
  • 示例:将一个大型数据库拆分为多个独立的部分,每个部分由不同的线程和锁管理。

2.无锁设计

  • 定义:通过无锁编程(如使用原子操作)来完全避免锁。
  • 示例:使用Java的AtomicInteger类进行计数器操作。

3.锁合并

  • 定义:在某些情况下,将多个锁合并为一个锁,以简化锁管理。
  • 示例:如果两个资源总是一起被访问,可以用一个锁来保护它们。

五、Demo讲解

package com.ctb.demo;/*** 关键字synchronized取得的锁都是对象锁,而不是把一段代码(方法)当做锁* 所以代码中哪个线程先执行synchronized关键字的方法,哪个线程就持有该方法所属对象的锁(Lock),* * 在静态方法上加synchronized关键字,表示锁定.class类,类一级别的锁(独占.class类)* * @author biao** 2024年*/
public class MyThread2 {private int num =0;public synchronized void printNum(String tag) {try {if (tag.equals("a")) {num=100;System.out.println("tag a,set num over!");Thread.sleep(1000);}else {num = 200;System.out.println("tag b,set num over!");}System.out.println("tag" + tag + "," + "num" + num);} catch (Exception e) {e.printStackTrace();}}public static void main(String[] args) {final MyThread2 m1 = new MyThread2();final MyThread2 m2 = new MyThread2();Thread t1 = new Thread(new Runnable() {@Overridepublic void run() {m1.printNum("a");}});Thread t2 = new Thread(new Runnable() {@Overridepublic void run() {m2.printNum("b");}});t1.start();t2.start();}}

结果:

package com.ctb.demo;/*** 关键字synchronized取得的锁都是对象锁,而不是把一段代码(方法)当做锁* 所以代码中哪个线程先执行synchronized关键字的方法,哪个线程就持有该方法所属对象的锁(Lock),* * 在静态方法上加synchronized关键字,表示锁定.class类,类一级别的锁(独占.class类)* * @author biao** 2024年2月28日-上午12:07:26*/
public class MyThread2 {private static int num =0;//	staticpublic static synchronized void printNum(String tag) {try {if (tag.equals("a")) {num=100;System.out.println("tag a,set num over!");Thread.sleep(1000);}else {num = 200;System.out.println("tag b,set num over!");}System.out.println("tag" + tag + "," + "num" + num);} catch (Exception e) {e.printStackTrace();}}public static void main(String[] args) {final MyThread2 m1 = new MyThread2();final MyThread2 m2 = new MyThread2();Thread t1 = new Thread(new Runnable() {@Overridepublic void run() {m1.printNum("a");}});Thread t2 = new Thread(new Runnable() {@Overridepublic void run() {m2.printNum("b");}});t1.start();t2.start();}}

结果:

总结:

关键字synchronized取得的锁都是对象锁,而不是把一段代码(方法)当做锁

  • 所以代码中哪个线程先执行synchronized关键字的方法,哪个线程就持有该方法所属对象的锁(Lock),

  • 在静态方法上加synchronized关键字,表示锁定.class类,类一级别的锁(独占.class类)

相关文章:

  • Python pandas openpyxl excel合并单元格,设置边框,背景色
  • 在 Linux 系统上安装 Android NDK
  • 呼叫中心系统的国产化替代方案
  • 【Flutter】 TextField限制长度时, 第三方手写输入法、ios原始拼音输入法输入被吞问题
  • swift微调牧歌数据电商多模态大语言模型
  • 【Spring6】1-12章源码级深入详解 IoC
  • 网络安全(补充)
  • 外卖抢单神器
  • 重学java 66.IO流 转换流
  • Linux-笔记 设备树插件
  • 3072. 将元素分配到两个数组中 II Rust 线段树 + 离散化
  • GIGE 协议摘录 —— GVSP 协议(三)
  • Web前端ES6-ES13笔记合集(下)
  • 【ARFoundation自学05】人脸追踪(AR Face manager)实现
  • 力扣1146.快照数组
  • 【前端学习】-粗谈选择器
  • 2017 年终总结 —— 在路上
  • Apache的80端口被占用以及访问时报错403
  • - C#编程大幅提高OUTLOOK的邮件搜索能力!
  • Javascript编码规范
  • quasar-framework cnodejs社区
  • vue2.0项目引入element-ui
  • 成为一名优秀的Developer的书单
  • 经典排序算法及其 Java 实现
  • 力扣(LeetCode)56
  • 事件委托的小应用
  • 写给高年级小学生看的《Bash 指南》
  • 找一份好的前端工作,起点很重要
  • 《码出高效》学习笔记与书中错误记录
  • ​520就是要宠粉,你的心头书我买单
  • ​configparser --- 配置文件解析器​
  • ​如何在iOS手机上查看应用日志
  • !!Dom4j 学习笔记
  • # 日期待t_最值得等的SUV奥迪Q9:空间比MPV还大,或搭4.0T,香
  • #我与Java虚拟机的故事#连载10: 如何在阿里、腾讯、百度、及字节跳动等公司面试中脱颖而出...
  • (04)Hive的相关概念——order by 、sort by、distribute by 、cluster by
  • (33)STM32——485实验笔记
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (待修改)PyG安装步骤
  • (附表设计)不是我吹!超级全面的权限系统设计方案面世了
  • (附源码)apringboot计算机专业大学生就业指南 毕业设计061355
  • (附源码)springboot人体健康检测微信小程序 毕业设计 012142
  • (入门自用)--C++--抽象类--多态原理--虚表--1020
  • (数位dp) 算法竞赛入门到进阶 书本题集
  • (算法二)滑动窗口
  • (一)Kafka 安全之使用 SASL 进行身份验证 —— JAAS 配置、SASL 配置
  • (转)清华学霸演讲稿:永远不要说你已经尽力了
  • .form文件_一篇文章学会文件上传
  • .NET Framework 的 bug?try-catch-when 中如果 when 语句抛出异常,程序将彻底崩溃
  • @GetMapping和@RequestMapping的区别
  • [ 代码审计篇 ] 代码审计案例详解(一) SQL注入代码审计案例
  • [ 隧道技术 ] 反弹shell的集中常见方式(二)bash反弹shell
  • [.net]官方水晶报表的使用以演示下载
  • [C++]AVL树怎么转
  • [C++基础]-初识模板