Java中到底有哪些锁

简介: 【9月更文挑战第24天】在Java中,锁主要分为乐观锁与悲观锁、自旋锁与自适应自旋锁、公平锁与非公平锁、可重入锁以及独享锁与共享锁。乐观锁适用于读多写少场景,通过版本号或CAS算法实现;悲观锁适用于写多读少场景,通过加锁保证数据一致性。自旋锁与自适应自旋锁通过循环等待减少线程挂起和恢复的开销,适用于锁持有时间短的场景。公平锁按请求顺序获取锁,适合等待敏感场景;非公平锁性能更高,适合频繁加解锁场景。可重入锁支持同一线程多次获取,避免死锁;独享锁与共享锁分别用于独占和并发读场景。

在 Java 中主要有以下几种锁:


一、乐观锁与悲观锁


  1. 悲观锁:
  • 概念:总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁。
  • 实现方式:传统的关系型数据库中的行锁、表锁以及 Java 中的同步代码块(使用synchronized关键字)和ReentrantLock等都是悲观锁的实现。
  • 适用场景:适合写操作多的场景,因为写操作具有排他性,采用悲观锁可以避免数据冲突。
  1. 乐观锁:
  • 概念:总是假设最好的情况,每次去拿数据的时候都认为别人不会修改,所以不会上锁。但是在更新的时候会判断在此期间别人有没有去更新这个数据,可以使用版本号机制和 CAS 算法实现。
  • 实现方式:在 Java 中可以使用AtomicInteger等原子类,它们是通过 CAS(Compare and Swap)算法实现的乐观锁。例如AtomicIntegerincrementAndGet()方法就是先比较当前值和预期值是否相等,如果相等则更新为新值并返回新值,否则不断重试直到成功。
  • 适用场景:适合读操作多的场景,因为读操作不会产生冲突,采用乐观锁可以提高并发性能。


二、自旋锁与自适应自旋锁


  1. 自旋锁:
  • 概念:当一个线程尝试获取锁的时候,如果锁已经被其他线程占用,那么该线程将循环等待,而不是立即挂起,看持有锁的线程是否会很快释放锁。
  • 实现方式:在 Java 中,自旋锁是通过循环和 CAS 操作实现的。例如,在AtomicReferencecompareAndSet()方法中,如果当前值与预期值不相等,则会不断循环尝试更新,这个过程就类似于自旋锁。
  • 适用场景:自旋锁适用于锁占用时间很短的场景,因为如果锁占用时间很长,那么自旋会浪费大量的 CPU 时间。
  1. 自适应自旋锁:
  • 概念:自适应自旋锁是在自旋锁的基础上,让自旋的时间不再固定,而是由前一次在同一个锁上的自旋时间及锁的拥有者的状态来决定。如果在同一个锁对象上,自旋等待刚刚成功获得过锁,并且持有锁的线程正在运行中,那么虚拟机就会认为这次自旋也很有可能再次成功,进而它将允许自旋等待持续相对更长的时间。如果对于某个锁,自旋很少成功获得过,那在以后要获取这个锁时将可能省略掉自旋过程,以避免浪费处理器资源。
  • 实现方式:在 Java 虚拟机中实现,对自旋锁的优化自动进行,开发者无需手动干预。


三、公平锁与非公平锁


  1. 公平锁:
  • 概念:多个线程按照申请锁的顺序来获取锁,线程直接进入队列中排队,队列中的第一个线程才能获得锁。
  • 实现方式:在 Java 中,可以通过构造ReentrantLock时传入true来创建公平锁,如ReentrantLock fairLock = new ReentrantLock(true);
  • 适用场景:公平锁适用于对线程等待时间敏感的场景,确保所有线程都有公平的机会获取锁,避免某些线程长时间等待。
  1. 非公平锁:
  • 概念:多个线程获取锁的顺序并不是按照申请锁的顺序,有可能后申请的线程比先申请的线程优先获取锁。在非公平锁的情况下,当一个线程释放锁时,会从等待队列中随机选择一个线程来获取锁,或者直接尝试获取锁,如果成功则直接获取锁,而不是按照先来后到的顺序。
  • 实现方式:ReentrantLocksynchronized都是默认的非公平锁。
  • 适用场景:非公平锁的性能通常比公平锁高,因为它减少了线程切换的开销。在大多数情况下,非公平锁可以提供更好的性能,特别是在锁被频繁获取和释放的情况下。


四、可重入锁(也叫递归锁)


  1. 概念:可重入锁是指同一个线程在外层方法获取锁的时候,在进入内层方法会自动获取锁。也就是说,线程可以进入任何一个它已经拥有的锁所同步着的代码块。
  2. 实现方式:synchronizedReentrantLock都是可重入锁。例如,一个使用synchronized修饰的方法内部又调用了另一个synchronized修饰的方法,同一个线程在调用外层方法获取锁后,进入内层方法时无需再次获取锁。
  3. 适用场景:在很多需要递归调用的场景中非常有用,避免了死锁的发生。


五、独享锁与共享锁


  1. 独享锁:
  • 概念:也叫排他锁,指的是锁一次只能被一个线程所持有。
  • 实现方式:synchronizedReentrantLock都是独享锁的实现。当一个线程获取到独享锁后,其他线程必须等待该线程释放锁后才能获取锁。
  • 适用场景:适用于对资源进行独占性访问的场景。
  1. 共享锁:
  • 概念:指的是锁可以被多个线程所持有。
  • 实现方式:在 Java 中,ReadWriteLock中的读锁是共享锁的实现。多个线程可以同时获取读锁来读取共享资源,但是在写锁被获取时,所有的读锁和写锁都会被阻塞,直到写锁被释放。
  • 适用场景:适用于对资源进行并发读操作较多的场景,可以提高读操作的并发性能。
相关文章
|
1月前
|
存储 Oracle 安全
揭秘Java并发核心:深入Hotspot源码腹地,彻底剖析Synchronized关键字的锁机制与实现奥秘!
【8月更文挑战第4天】在Java并发世界里,`Synchronized`如同导航明灯,确保多线程环境下的代码安全执行。它通过修饰方法或代码块实现独占访问。在Hotspot JVM中,`Synchronized`依靠对象监视器(Object Monitor)机制实现,利用对象头的Mark Word管理锁状态。
40 1
|
13天前
|
存储 缓存 安全
【Java面试题汇总】多线程、JUC、锁篇(2023版)
线程和进程的区别、CAS的ABA问题、AQS、哪些地方使用了CAS、怎么保证线程安全、线程同步方式、synchronized的用法及原理、Lock、volatile、线程的六个状态、ThreadLocal、线程通信方式、创建方式、两种创建线程池的方法、线程池设置合适的线程数、线程安全的集合?ConcurrentHashMap、JUC
【Java面试题汇总】多线程、JUC、锁篇(2023版)
|
9天前
|
Java 数据库
JAVA并发编程-一文看懂全部锁机制
曾几何时,面试官问:java都有哪些锁?小白,一脸无辜:用过的有synchronized,其他不清楚。面试官:回去等通知! 今天我们庖丁解牛说说,各种锁有什么区别、什么场景可以用,通俗直白的分析,让小白再也不怕面试官八股文拷打。
|
9天前
|
安全 Java 开发者
Java并发编程中的锁机制解析
本文深入探讨了Java中用于管理多线程同步的关键工具——锁机制。通过分析synchronized关键字和ReentrantLock类等核心概念,揭示了它们在构建线程安全应用中的重要性。同时,文章还讨论了锁机制的高级特性,如公平性、类锁和对象锁的区别,以及锁的优化技术如锁粗化和锁消除。此外,指出了在高并发环境下锁竞争可能导致的问题,并提出了减少锁持有时间和使用无锁编程等策略来优化性能的建议。最后,强调了理解和正确使用Java锁机制对于开发高效、可靠并发应用程序的重要性。
15 3
|
29天前
|
存储 Java
Java锁是什么?简单了解
在高并发环境下,锁是Java中至关重要的概念。锁或互斥是一种同步机制,用于限制多线程环境下的资源访问,确保排他性和并发控制。例如,超市储物柜仅能存放一个物品,若三人同时使用,则需通过锁机制确保每次只有一个线程访问。Java中可以通过`synchronized`关键字实现加锁,确保关键代码段的原子性,避免数据不一致问题。正确使用锁可有效提升程序的稳定性和安全性。
Java锁是什么?简单了解
|
30天前
|
小程序 Java 开发工具
【Java】@Transactional事务套着ReentrantLock锁,锁竟然失效超卖了
本文通过一个生动的例子,探讨了Java中加锁仍可能出现超卖问题的原因及解决方案。作者“JavaDog程序狗”通过模拟空调租赁场景,详细解析了超卖现象及其背后的多线程并发问题。文章介绍了四种解决超卖的方法:乐观锁、悲观锁、分布式锁以及代码级锁,并重点讨论了ReentrantLock的使用。此外,还分析了事务套锁失效的原因及解决办法,强调了事务边界的重要性。
54 2
【Java】@Transactional事务套着ReentrantLock锁,锁竟然失效超卖了
|
21天前
|
Oracle Java 关系型数据库
【颠覆性升级】JDK 22:超级构造器与区域锁,重塑Java编程的两大基石!
【9月更文挑战第6天】JDK 22的发布标志着Java编程语言在性能和灵活性方面迈出了重要的一步。超级构造器和区域锁这两大基石的引入,不仅简化了代码设计,提高了开发效率,还优化了垃圾收集器的性能,降低了应用延迟。这些改进不仅展示了Oracle在Java生态系统中的持续改进和创新精神,也为广大Java开发者提供了更多的可能性和便利。我们有理由相信,在未来的Java编程中,这些新特性将发挥越来越重要的作用,推动Java技术不断向前发展。
|
1月前
|
Java 开发者
Java多线程教程:使用ReentrantLock实现高级锁功能
Java多线程教程:使用ReentrantLock实现高级锁功能
28 1
|
1月前
|
安全 Java 开发者
【锁的艺术】StampedLock:Java并发编程的新武器!
【8月更文挑战第24天】`StampedLock`, 作为 Java 8 引入的新特性,为开发者提供了一种相较于传统 `ReentrantReadWriteLock` 更高效且灵活的锁机制。它属于 `java.util.concurrent.locks` 包,主要特点包括乐观读锁,这在多读少写的场景下能显著提升性能。
35 1
|
1月前
|
存储 SQL 关系型数据库
深入MySQL锁机制:原理、死锁解决及Java防范技巧
深入MySQL锁机制:原理、死锁解决及Java防范技巧