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

Java多线程:synchronized方法独占锁的秘密

文章目录

  • Java多线程:synchronized方法独占锁的秘密?
    • 什么是synchronized?
    • synchronized背后的独占锁机制
      • 1. 独占锁的定义
      • 2. 锁的粒度
      • 3. 锁的获取与释放
    • 独占锁的优缺点
      • 优点
      • 缺点
    • 案例分析:synchronized方法的常见问题
      • 例子代码:
      • 问题分析
      • 解决方案
        • 方法一:细化锁粒度
        • 方法二:使用更高效的锁机制
    • 总结
    • 通过合理使用`synchronized`方法和其他并发工具(如`ReentrantLock`),我们可以写出既安全又高效的多线程代码。
      • 📚 领取 | 1000+ 套高质量面试题大合集(无套路,闫工带你飞一把)!

Java多线程:synchronized方法独占锁的秘密?

大家好,欢迎来到闫工的博客!今天我们要聊的是一个非常经典的话题——Java多线程中的synchronized关键字以及它背后的“独占锁”机制。作为一个Java开发者,你可能每天都在用synchronized,但你真的了解它的秘密吗?别急,咱们一步步来拆解这个知识点。


什么是synchronized?

首先,synchronized是Java中用来实现线程同步的一个关键字。简单来说,它用于确保在多线程环境下,同一时间只有一个线程可以执行某个方法或代码块。这样就能避免多个线程同时修改共享资源时出现的竞态条件(Race Condition)和不一致的问题。

举个栗子,假设你有一个银行账户,余额是100元。如果有两个线程同时尝试从这个账户中取钱,可能会发生以下情况:

  • 线程A读取到余额是100元,并决定取出50元。
  • 同时,线程B也读取到余额是100元,并取出30元。

这样就会导致最终的余额变成20元(100 - 50 - 30),但实际上这是错误的,因为两个线程都操作了同一个资源。为了避免这种情况,synchronized可以帮助我们实现“独占锁”,确保同一时间只有一个线程可以操作这个资源。


synchronized背后的独占锁机制

那么问题来了,synchronized是如何实现这种“独占锁”的呢?让我们从底层原理来理解这个问题。

1. 独占锁的定义

在Java中,“独占锁”指的是在同一时间只有一个线程可以持有该锁并执行被保护的方法或代码块。当一个线程进入synchronized方法时,它会获取到这把“锁”,其他尝试进入这个方法的线程必须等待,直到当前线程释放锁。

2. 锁的粒度

在Java中,synchronized可以修饰两种类型的代码:实例方法静态方法。它们分别对应不同的锁粒度:

  • 实例方法:锁是针对实例对象的。也就是说,每个实例对象都有自己的锁。
  • 静态方法:锁是针对类的。无论有多少个实例对象,所有线程在调用同一个类的静态synchronized方法时都会共享同一把锁。

举个例子:

publicclassAccount{privatedoublebalance;// 实例synchronized方法publicsynchronizedvoidwithdraw(doubleamount){if(balance>=amount){balance-=amount;}}// 静态synchronized方法publicstaticsynchronizedvoidtransfer(Accountfrom,Accountto,doubleamount){from.withdraw(amount);to.deposit(amount);}}

在这个例子中:

  • withdraw方法是实例方法,每个Account对象都有自己的锁。
  • transfer方法是静态方法,所有线程在调用这个方法时都会争夺同一个类级别的锁。

3. 锁的获取与释放

当一个线程进入synchronized方法时,它会尝试获取对应的锁。如果锁没有被其他线程持有,那么当前线程就会成功获取锁并执行方法;否则,该线程会被阻塞,直到锁被释放。

释放锁的过程发生在以下几种情况:

  • 当前线程正常退出synchronized方法。
  • 当前线程在synchronized方法中抛出异常。
  • 当前线程调用了wait()方法并释放了锁(后面会详细讲解)。

独占锁的优缺点

了解了独占锁的基本原理后,我们再来分析它的优缺点。

优点

  1. 简单易用synchronized关键字内置在Java语言中,使用起来非常方便。
  2. 自动管理:Java虚拟机(JVM)会自动管理锁的获取与释放,开发者不需要手动实现复杂的锁机制。
  3. 可重入性:一个线程可以多次进入同一个synchronized方法而不会被阻塞。

缺点

  1. 粒度较粗synchronized通常会对整个方法进行加锁,这可能会导致不必要的性能开销。如果只是其中一小部分代码需要同步,使用synchronized可能会显得过于“保守”。
  2. 死锁风险:多个线程在争夺多把锁时,可能会发生死锁(Deadlock)。例如,线程A持有锁1并试图获取锁2,而线程B持有锁2并试图获取锁1,就会导致双方都无法继续执行。

案例分析:synchronized方法的常见问题

为了更好地理解synchronized的实际应用和潜在问题,我们来看一个经典的案例——银行转账。

例子代码:

publicclassBank{publicsynchronizedvoidtransfer(Accountfrom,Accountto,doubleamount){if(from.getBalance()>=amount){from.withdraw(amount);// 模拟网络延迟try{Thread.sleep(1000);}catch(InterruptedExceptione){e.printStackTrace();}to.deposit(amount);}}publicstaticvoidmain(String[]args){Bankbank=newBank();Accountalice=newAccount("Alice",100.0);Accountbob=newAccount("Bob",50.0);// 创建两个线程同时执行转账Threadthread1=newThread(()->{bank.transfer(alice,bob,30.0);});Threadthread2=newThread(()->{bank.transfer(bob,alice,40.0);});thread1.start();thread2.start();}}classAccount{privateStringname;privatedoublebalance;publicAccount(Stringname,doublebalance){this.name=name;this.balance=balance;}publicsynchronizedvoidwithdraw(doubleamount){if(balance>=amount){balance-=amount;System.out.println(name+" withdrew: "+amount);}}publicsynchronizedvoiddeposit(doubleamount){balance+=amount;System.out.println(name+" deposited: "+amount);}publicdoublegetBalance(){returnbalance;}}

问题分析

在上面的代码中,Bank类中的transfer方法是synchronized的,这意味着每次只能有一个线程执行转账操作。然而,这个实现仍然存在一些潜在的问题:

  1. 粒度问题transfer方法对整个转账过程进行了加锁,但实际上只需要对账户的操作进行同步。
  2. 性能瓶颈:如果多个线程频繁调用transfer方法,由于独占锁的存在,可能会导致排队和性能下降。

解决方案

为了优化这个问题,我们可以考虑以下几种方法:

方法一:细化锁粒度

将锁粒度从整个transfer方法降低到具体的账户操作。例如,在Account类中对每个操作进行加锁。

修改后的代码:

publicclassBank{publicvoidtransfer(Accountfrom,Accountto,doubleamount){if(from.getBalance()>=amount){synchronized(from){from.withdraw(amount);}try{Thread.sleep(1000);}catch(InterruptedExceptione){e.printStackTrace();}synchronized(to){to.deposit(amount);}}}}
方法二:使用更高效的锁机制

在Java中,除了内置的synchronized关键字,还可以使用ReentrantLock来实现更灵活和高效的锁管理。

修改后的代码:

importjava.util.concurrent.locks.ReentrantLock;publicclassBank{privatefinalReentrantLocklock=newReentrantLock();publicvoidtransfer(Accountfrom,Accountto,doubleamount){if(from.getBalance()>=amount){try{// 尝试获取锁,参数为true表示阻塞等待lock.lock();from.withdraw(amount);Thread.sleep(1000);to.deposit(amount);}catch(InterruptedExceptione){e.printStackTrace();}finally{// 释放锁lock.unlock();}}}publicstaticvoidmain(String[]args){Bankbank=newBank();Accountalice=newAccount("Alice",100.0);Accountbob=newAccount("Bob",50.0);Threadthread1=newThread(()->{bank.transfer(alice,bob,30.0);});Threadthread2=newThread(()->{bank.transfer(bob,alice,40.0);});thread1.start();thread2.start();}}

总结

synchronized关键字是Java中实现线程同步的一种简单而有效的方式,但它也有一些局限性。在实际开发中,我们需要根据具体场景选择合适的锁粒度和锁机制,以避免性能瓶颈和潜在的死锁问题。

通过合理使用synchronized方法和其他并发工具(如ReentrantLock),我们可以写出既安全又高效的多线程代码。

📚 领取 | 1000+ 套高质量面试题大合集(无套路,闫工带你飞一把)!

成体系的面试题,无论你是大佬还是小白,都需要一套JAVA体系的面试题,我已经上岸了!你也想上岸吗?

闫工精心准备了程序准备面试?想系统提升技术实力?闫工精心整理了1000+ 套涵盖前端、后端、算法、数据库、操作系统、网络、设计模式等方向的面试真题 + 详细解析,并附赠高频考点总结、简历模板、面经合集等实用资料!

✅ 覆盖大厂高频题型
✅ 按知识点分类,查漏补缺超方便
✅ 持续更新,助你拿下心仪 Offer!

📥免费领取👉 点击这里获取资料

已帮助数千位开发者成功上岸,下一个就是你!✨

http://www.jsqmd.com/news/324918/

相关文章:

  • 2026年评价高的防沉触变剂/防沉剂抗流挂触变剂厂家选购指南与推荐
  • 2026年质量好的精密冷拉异型钢/冷拉异型钢六角钢实力厂家TOP推荐榜
  • 2026年比较好的注塑磁铁/粘结钕铁硼塑磁厂家推荐及选择指南
  • 天津资深大平层设计师|塑造松弛奢适居住感[特殊字符]
  • 2026年热门的卡通布牛津布/磨砂布牛津布行业内口碑厂家排行榜
  • es-api介绍 - 详解
  • 2026年质量好的纳米材料分散剂/高分子分散剂厂家最新热销排行
  • 2024信奥赛C++提高组csp-s复赛真题及题解:决斗
  • 2026年热门的工业空冷器/空冷器管束厂家最新用户好评榜
  • 版权信息要保留?开源项目使用规范
  • 互联网大厂Java面试:从分布式缓存到消息队列的技术场景解析
  • GLM-Image WebUI启动故障排查:加载失败/显存不足/依赖缺失解决方案
  • 2026年比较好的高鱼粉含量鲈鱼饲料/发酵蛋白鲈鱼饲料品牌推荐榜
  • 警告!你的RAG系统正在裸奔!USENIX Security最新论文揭示90%成功率攻击手法,附防御方案
  • 2026年评价高的网带式抛丸机/通过式抛丸机用户好评厂家排行
  • 2026年质量好的热镀锌电缆桥架/定制电缆桥架TOP实力厂家推荐榜
  • 2026年知名的酸洗冷轧带钢/黑退冷轧带钢厂家实力及用户口碑排行榜
  • 2026年口碑好的涂塑支架/支架厂家最新TOP实力排行
  • 2026年口碑好的60孔催化剂/船催化剂厂家最新实力排行
  • 2026年热门的立式排污泵/耐高温排污泵最新TOP品牌厂家排行
  • 2026年知名的化工离心泵/活鱼输送离心泵厂家推荐及采购指南
  • 2026年评价高的闭式循环水冷却塔/工业冷却塔用户好评厂家排行
  • 基于SpringBoot的建筑工程项目管理系统(源码+lw+部署文档+讲解等)
  • 2026年知名的胶辊/木业胶辊实力厂家TOP推荐榜
  • 基于SpringBoot的大连市IT行业招聘平台的设计与实现(源码+lw+部署文档+讲解等)
  • 2026年评价高的中温脱硝催化剂/蜂窝式脱硝催化剂厂家最新TOP排行榜
  • GPEN能修复侧脸和遮挡人脸吗?实测结果来了
  • 基于SpringBoot的定制化设计服务平台系统(源码+lw+部署文档+讲解等)
  • FusionGraphNet-Pro:基于时空图神经网络的工业设备故障诊断(Python)
  • Clawdbot汉化版企业实操:制造业设备报修微信群AI自动分派工单