Java锁之舞:性能分析与优化之路

目录

一、同步锁性能分析

(一)性能验证说明

1. 使用同步锁的代码示例

2. 不使用同步锁的代码示例

3. 结果与讨论

(二)案例初步优化分析说明

1. 使用AtomicInteger原子类尝试优化分析

2. 对AtomicInteger原子类进一步优化

3. 结论说明(LongAdder原理理解体会)

二、回顾Java锁优化

(一)synchronized 关键字

1. monitor 锁的实现原理

2.分级锁

偏向锁(Biased Locking)

轻量级锁

重量级锁

3. 锁升级一览

(二)concurrent 包里面的 Lock

1. 锁机制基于线程而不是基于调用(可重入锁)

2. Lock 主要方法

lock()

unlock()

tryLock()

tryLock(long timeout, TimeUnit unit)

lockInterruptibly()

3. 读写锁ReentrantReadWriteLock

基本内容说明

性能验证说明

4.乐观读取、悲观读取和写入的机制:StampedLock

基本内容说明

性能验证说明

5. 公平锁与非公平锁

synchronized关键字 vs Lock接口

功能验证

(三)Java 中两种加锁方式对比和建议

三、锁的优化手段

(一)减少锁的粒度

(二)减少锁持有时间

(三)锁分级

(四)锁分离

(五)锁消除

(六)乐观锁

(七)无锁

参考文章


在多线程编程中,锁是保证线程安全的重要手段之一,但如何选择合适的锁并进行优化,一直是我们面临的挑战。本博客探讨Java中同步锁的性能分析与优化之路,从使用同步锁和不使用同步锁的性能对比入手,逐步展开对锁的优化手段和技术原理的解析,帮助读者更好地理解和应用Java中的锁机制。

一、同步锁性能分析

同步锁在多线程编程中是保证线程安全的重要工具,其性能开销一直是不可忽视的存在。

(一)性能验证说明

为了直观说明我们可以直接先准备两个Java代码用例,我们通过高并发环境下的计数器递增操作来对比使用同步锁和不使用同步锁的性能差异。

1. 使用同步锁的代码示例

使用ReentrantLock来保护对共享资源(counter)的访问,确保同一时间只有一个线程可以对计数器进行操作。具体代码如下:

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 使用了ReentrantLock来保护对共享资源(counter)的访问,确保同一时间只有一个线程可以对计数器进行操作。* @author: zhangyanfeng* @create: 2024-06-05 22:54**/
public class SyncLockExample {private static int counter = 0;private static final ReentrantLock lock = new ReentrantLock();public static void main(String[] args) throws InterruptedException {long startTime = System.currentTimeMillis();Thread[] threads = new Thread[100];for (int i = 0; i < 100; i++) {threads[i] = new Thread(new IncrementWithLock());threads[i].start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Time with lock: " + (endTime - startTime) + " ms");}static class IncrementWithLock implements Runnable {@Overridepublic void run() {for (int i = 0; i < 1000000; i++) {lock.lock();try {counter++;} finally {lock.unlock();}}}}
}

2. 不使用同步锁的代码示例

不使用任何同步机制,直接操作共享资源。具体代码如下:

package org.zyf.javabasic.thread.lock.opti;/*** @program: zyfboot-javabasic* @description: 不使用任何同步机制,直接操作共享资源。* @author: zhangyanfeng* @create: 2024-06-05 22:55**/
public class NoSyncLockExample {private static int counter = 0;public static void main(String[] args) throws InterruptedException {long startTime = System.currentTimeMillis();Thread[] threads = new Thread[100];for (int i = 0; i < 100; i++) {threads[i] = new Thread(new IncrementWithoutLock());threads[i].start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Time without lock: " + (endTime - startTime) + " ms");}static class IncrementWithoutLock implements Runnable {@Overridepublic void run() {for (int i = 0; i < 1000000; i++) {counter++;}}}
}

3. 结果与讨论

运行以上代码,我当前的机器上可以直观的看到

  • 使用同步锁的时间: 1314 ms
  • 不使用同步锁的时间: 20 ms

从结果中可以明显看出,同步锁会带来显著的性能开销。同步锁的存在增加了线程间的等待时间和上下文切换的开销,从而降低了程序的整体运行效率。所以在使用锁时,对锁的优化使用是必不可少的。

(二)案例初步优化分析说明

在开始讲解一些常用的优化手段的时候,我们先就现在这个用例来谈谈可能我们一般可以想到的直观优化手段。

1. 使用AtomicInteger原子类尝试优化分析

Java的java.util.concurrent.atomic包提供了一些原子类,可以在并发编程中避免显式加锁。最简单的我们可以使用AtomicInteger来替代显式的锁。

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.atomic.AtomicInteger;/*** @program: zyfboot-javabasic* @description: 使用AtomicInteger来替代显式的锁* @author: zhangyanfeng* @create: 2024-06-05 23:07**/
public class AtomicExample {private static AtomicInteger counter = new AtomicInteger(0);public static void main(String[] args) throws InterruptedException {long startTime = System.currentTimeMillis();Thread[] threads = new Thread[100];for (int i = 0; i < 100; i++) {threads[i] = new Thread(new IncrementAtomic());threads[i].start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Time with AtomicInteger: " + (endTime - startTime) + " ms");}static class IncrementAtomic implements Runnable {@Overridepublic void run() {for (int i = 0; i < 1000000; i++) {counter.incrementAndGet();}}}
}

理论上这样优化后性能上必会上升,但实际上运行后其耗时为 6714 ms,性能反而变差了。这里其实我们之前的博客中也有讲过,其主要的原因有两个:

  • 原子类的开销AtomicIntegerincrementAndGet方法虽然是无锁的,但它依赖于底层的CAS(Compare-And-Swap)操作。CAS操作虽然是无锁的,但在高并发情况下,多个线程同时尝试更新同一个变量时,CAS操作可能会频繁地失败并重试,从而导致性能下降。相比之下,ReentrantLock在某些情况下可能反而表现更好,尤其是在锁争用不是特别激烈的时候。
  • 高并发下的内存争用在高并发情况下,多个线程同时访问和修改共享变量会导致内存争用。这种争用在使用AtomicInteger时表现得更为明显,因为每次操作都需要与主内存同步,可能会导致缓存一致性协议的开销。

2. 对AtomicInteger原子类进一步优化

我们可以尝试以下方法来进一步优化:

  • 减少线程数量:在本示例中,我们使用了100个线程同时访问共享变量,这可能导致过多的上下文切换和争用。可以尝试减少线程数量,看看性能是否有所改善。
  • 使用更高效的同步机制:可以尝试使用其他同步机制,如LongAdder或ConcurrentLinkedQueue等,这些工具在高并发场景下通常表现更好。

这里我们直接用LongAdder验证,具体代码如下:

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.atomic.LongAdder;/*** @program: zyfboot-javabasic* @description: LongAdder在高并发情况下比AtomicInteger有更好的性能* @author: zhangyanfeng* @create: 2024-06-05 23:26**/
public class LongAdderExample {private static LongAdder counter = new LongAdder();public static void main(String[] args) throws InterruptedException {long startTime = System.currentTimeMillis();Thread[] threads = new Thread[100];for (int i = 0; i < 100; i++) {threads[i] = new Thread(new IncrementLongAdder());threads[i].start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Time with LongAdder: " + (endTime - startTime) + " ms");}static class IncrementLongAdder implements Runnable {@Overridepublic void run() {for (int i = 0; i < 1000000; i++) {counter.increment();}}}
}

运行后发现这个时候的耗时基本在204 ms,优化还是很明显的。

3. 结论说明(LongAdder原理理解体会)

在实际应用中,选择合适的优化方法需要根据具体的业务逻辑和并发需求进行权衡和调整。这里我们针对LongAdder的优化进行说明一下,它是基于了 CAS 分段锁的思想实现的。线程去读写一个 LongAdder 类型的变量时,流程如下:

基于 Unsafe 提供的 CAS 操作 +valitale 去实现的。在 LongAdder 的父类 Striped64 中维护着一个 base 变量和一个 cell 数组,当多个线程操作一个变量的时候,先会在这个 base 变量上进行 cas 操作,当它发现线程增多的时候,就会使用 cell 数组。比如当 base 将要更新的时候发现线程增多(也就是调用 casBase 方法更新 base 值失败),那么它会自动使用 cell 数组,每一个线程对应于一个 cell ,在每一个线程中对该 cell 进行 cas 操作,这样就可以将单一 value 的更新压力分担到多个 value 中去,降低单个 value 的 “热度”,同时也减少了大量线程的空转,提高并发效率,分散并发压力。这种分段锁需要额外维护一个内存空间 cells ,不过在高并发场景下,这点成本几乎可以忽略。

我觉得可以把 LongAdder 想象成一个超市收银台系统:

  • base 变量:一个主收银台,所有顾客最开始都会排队在这里付款。
  • cell 数组:多个备用收银台,当主收银台忙不过来时,顾客会被分配到不同的备用收银台去付款。

这样做的好处是,当有大量顾客(高并发)时,大家不会都挤在一个收银台前,避免了长时间的等待,提高了结账效率。

二、回顾Java锁优化

Java 中的 synchronized 关键字和 java.util.concurrent.locks 包中的 Lock 接口(如 ReentrantLock)是两种常见的加锁方式,它们各有优缺点,针对这两种锁,JDK 自身做了很多的优化,它们的实现方式也是不同的。我们不妨进行简单的回顾一下。

(一)synchronized 关键字

synchronized 关键字给代码或者方法上锁时,都有显示或者隐藏的上锁对象。当一个线程试图访问同步代码块时,它首先必须得到锁,而退出或抛出异常时必须释放锁(注意是 synchronized 关键字的内置机制,由 Java 语言和 JVM 自动管理。我们在使用 synchronized 关键字时,无需手动编写获取和释放锁的代码,synchronized 会自动处理这些细节)。

  • 给普通方法加锁时:锁定对象是实例对象 (this)。相同实例的 synchronized 方法之间是串行的,不同实例之间则不会互相影响。

  • 给静态方法加锁时:锁定对象是类的 Class 对象。所有实例共享同一个类锁,因此静态 synchronized 方法在所有实例上都是串行的。

  • 给代码块加锁时:可以指定任意对象作为锁。锁的粒度更细,可以针对不同的对象分别加锁,灵活控制并发。

synchronized 的实现依赖于 JVM 和操作系统的原语,如监视器锁和信号量。JVM 在执行同步块时,会插入相应的加锁和解锁指令。

1. monitor 锁的实现原理

synchronized 关键字在 Java 字节码中通过监视器(Monitor)指令来实现。具体来说,当 Java 编译器编译包含 synchronized 关键字的代码时,会在相应的位置插入 monitorentermonitorexit 指令。JVM 在运行这些字节码时,会负责管理锁的获取和释放。

  • 对于同步实例方法,编译器在字节码中并不会显式插入 monitorentermonitorexit 指令。相反,它会在方法的访问标志(access flag)中添加 ACC_SYNCHRONIZED 标志。JVM 在调用该方法时会自动获取实例的监视器锁,并在方法返回或抛出异常时自动释放锁。
  • 对于同步静态方法,编译器同样会在方法的访问标志中添加 ACC_SYNCHRONIZED 标志。JVM 在调用该方法时会自动获取类的监视器锁,并在方法返回或抛出异常时自动释放锁。
  • 对于同步代码块,编译器会在进入同步块时插入 monitorenter 指令,在退出同步块时插入 monitorexit 指令。这些指令用于显式地获取和释放指定的锁对象。

现在我们直观的验证一下:

package org.zyf.javabasic.thread.lock.opti;/*** @program: zyfboot-javabasic* @description: 同步方法和同步代码块的示例类,以及它们在字节码中的表现。* @author: zhangyanfeng* @create: 2024-06-06 07:56**/
public class SynchronizedExample {private int count = 0;private final Object lock = new Object();// 同步实例方法public synchronized void increment() {count++;}// 同步静态方法public static synchronized void staticIncrement() {// 静态方法体}// 同步代码块public void blockIncrement() {synchronized (lock) {count++;}}
}

可以使用 javap -c SynchronizedExample 命令查看字节码如下:

public synchronized void increment();flags: ACC_PUBLIC, ACC_SYNCHRONIZEDCode:stack=2, locals=1, args_size=10: aload_01: getfield      #2                  // Field count:I4: iconst_15: iadd6: aload_07: swap8: putfield      #2                  // Field count:I11: returnpublic static synchronized void staticIncrement();flags: ACC_PUBLIC, ACC_STATIC, ACC_SYNCHRONIZEDCode:stack=0, locals=0, args_size=00: returnpublic void blockIncrement();flags: ACC_PUBLICCode:stack=2, locals=3, args_size=10: aload_01: getfield      #3                  // Field lock:Ljava/lang/Object;4: dup5: astore_16: monitorenter7: aload_08: dup9: getfield      #2                  // Field count:I12: iconst_113: iadd14: putfield      #2                 // Field count:I17: aload_118: monitorexit19: goto          2722: astore_223: aload_124: monitorexit25: aload_226: athrow27: returnException table:from    to  target type7    19    22   any22    25    22   any

正如我们上面总结的一样,整体来说:

  • 使用 ACC_SYNCHRONIZED 标志,JVM 自动处理锁的获取和释放。
  • 使用 monitorentermonitorexit 指令显式处理锁的获取和释放,确保在正常退出和异常退出时都能正确释放锁。

这两者虽然显示效果不同,但他们都是通过 monitor 来实现同步的。当一个线程进入一个同步块或同步方法时,它会尝试获取与该对象关联的 Monitor。如果 Monitor 已经被其他线程持有,当前线程将会阻塞,直到 Monitor 被释放。

虽然具体实现可能会根据不同的 JVM 实现有所差异,但通常可以用以下示意来表示 Monitor 的结构:

Monitor {Thread owner;          // 当前持有锁的线程int recursionCount;    // 递归计数器List<Thread> entryList; // 进入列表,等待获取锁的线程队列List<Thread> waitSet;   // 等待列表,调用 wait() 方法等待的线程队列List<Thread> exitList;  // 退出列表,等待退出的线程队列(可能的实现)
}

锁相关主要的流程如下:

初始状态:多个线程尝试进入同步代码块或方法,所有线程首先进入 EntryList 队列。这些线程处于“Waiting for monitor entry”状态( jstack 命令可查看)。

获取锁的过程:某个线程成功获取 Monitor 后,_owner 变量设置为当前线程,表示该线程持有锁。Monitor_count 计数器自增 1,表示锁被持有的次数。

等待和释放锁:持有 Monitor 的线程调用 wait() 方法时,会释放锁,并进入 WaitSet 队列。释放锁后,_owner 变量恢复为 null_count 计数器自减 1。线程状态变为“in Object.wait()”,等待被其他线程唤醒。

执行完成释放锁:持有 Monitor 的线程执行完同步代码块或方法时,会释放锁。释放锁后,_owner 变量恢复为 null_count 计数器自减 1。

唤醒等待的线程:线程调用 notify()notifyAll() 方法时,会唤醒 WaitSet 中的一个或多个线程。被唤醒的线程重新进入 EntryList 队列,等待再次获取 Monitor 锁。

2.分级锁

在 JDK 1.8 中,synchronized 关键字的性能得到了显著提升,这主要得益于 JVM 对锁机制进行了一系列优化:锁的分级及其优化路径(大体可以按照下面的路径进行升级:偏向锁 — 轻量级锁 — 重量级锁,锁只能升级,不能降级,所以一旦升级为重量级锁,就只能依靠操作系统进行调度)。

要想了解锁升级的过程,需要先看一下对象在内存里的结构。

在 Java 中,对象的内存布局中包含了 MarkWordClass PointerInstance DataPadding 等部分。而锁的升级过程主要与对象头的 MarkWord 有关。

要知道MarkWord 是对象头中用于存储对象的运行时信息的。在 64 位 JVM 中,MarkWord 的长度为 64 位。在 32 位 JVM 中,MarkWord 的长度为 32 位(如上图)。

偏向锁(Biased Locking)
  • 单线程高效使用锁:在只有一个线程使用锁的情况下,偏向锁效率最高。
  • 获取锁:第一个线程访问同步块时,会检查对象头 Mark Word 的标志位 Tag 是否为 01。如果是,线程将自己的线程 ID 写入 Mark Word,锁进入偏向锁状态。
  • 撤销偏向锁:当其他线程尝试获取锁时,如果 Mark Word 中的线程 ID 不匹配,偏向锁会被撤销,升级为轻量级锁。
  • 适合单线程场景,高效。
轻量级锁
  • 自旋锁获取:参与竞争的线程会在自己的线程栈中生成一个 LockRecord (LR),通过 CAS(自旋)的方式,将锁对象头中的 Mark Word 设置为指向自己的 LR 的指针。设置成功的线程获得锁。
  • 自旋失败:如果自旋多次失败,锁会升级为重量级锁。
  • 适合短期竞争,自旋获取锁。
重量级锁
  • 系统调度:重量级锁会导致线程挂起,进入操作系统内核态,等待操作系统调度,然后再映射回用户态。系统调用的开销很高,锁的膨胀到重量级锁就意味着性能下降。
  • 激烈竞争:如果共享变量竞争激烈,锁会迅速膨胀为重量级锁。如果并发竞争严重,可以使用 -XX:-UseBiasedLocking 禁用偏向锁,可能会有一些性能提升。
  • 适合长期竞争,但性能开销大。

3. 锁升级一览

(二)concurrent 包里面的 Lock

synchronized 是 Java 提供的最基本的同步机制,通过简单易用的语法确保线程安全。然而,随着并发需求的复杂化,Java 的并发包 (java.util.concurrent) 提供了更多高级和高效的并发工具,如 ReentrantLockReadWriteLockAtomic 类等,来应对更复杂的并发场景。在实际开发中,应根据具体情况选择合适的同步机制。现在我们聚焦在Lock进行分析。

1. 锁机制基于线程而不是基于调用(可重入锁)

这种锁机制基于线程而不是基于调用” 的意思是说,当一个线程持有锁时,它可以在同一个线程的不同调用链中多次获取同一把锁,而不会被阻塞或引发死锁。这是因为锁是跟线程关联的,而不是跟调用栈关联的。

假设有一个对象 example,它有三个同步方法 abc,每个方法都被 synchronized 关键字修饰:

package org.zyf.javabasic.thread.lock.opti;/*** @program: zyfboot-javabasic* @description: 锁机制基于线程而不是基于调用* @author: zhangyanfeng* @create: 2024-06-07 19:04**/
public class LockExample {public synchronized void a() {System.out.println("In method a");b();  // 调用 b 方法}public synchronized void b() {System.out.println("In method b");c();  // 调用 c 方法}public synchronized void c() {System.out.println("In method c");}public static void main(String[] args) {LockExample example = new LockExample();example.a();}
}

main 方法中,我们调用了 example.a()。在 a 方法内部,又调用了 b 方法,而 b 方法内部又调用了 c 方法。这种调用关系如下:

example.a()-> example.b()-> example.c()

当线程调用 example.a() 时,由于 a 方法是同步方法,线程必须先获得对象 example 的锁。在 a 方法内部,线程调用 example.b()。虽然 b 方法也是同步方法,但是由于当前线程已经持有了 example 对象的锁,所以它可以继续执行,不需要再次获取锁。类似地,在 b 方法内部,线程调用 example.c() 时,也不需要再次获取锁。

这个过程中锁是跟线程关联的,而不是跟每次调用关联的。一个线程持有锁之后,可以在它的调用栈中多次获取同一把锁,而不需要重新获取锁,也不会被阻塞。

如果 Java 的锁机制不是基于线程的,而是基于每次调用的,那么在上面的示例中,线程在调用 b 方法时会尝试再次获取 example 对象的锁,但是由于它已经持有这个锁,这将导致死锁。因此,基于线程的锁机制(即可重入锁)避免了这种情况,使得一个线程在持有锁时,可以多次获取同一把锁。Java 的 ReentrantLock 类也支持可重入性,将以上synchronized替换成其效果也是一样的。

像上面这样,在并发编程中,可重入锁(Reentrant Lock)指的是一个线程可以多次获得同一把锁。可重入锁的作用在于避免线程死锁,当一个线程已经持有了一个锁,再次请求该锁时可以直接获取,而无需再次等待。这种锁机制基于线程而不是基于调用。

2. Lock 主要方法

在 Java 的并发编程中,Lock 接口提供了比 synchronized 更加灵活和强大的锁机制。Lock 与 synchronized 的使用方法不同,它需要手动加锁,然后在 finally 中解锁。

Lock 接口是基于 AQS(AbstractQueuedSynchronizer)实现的,而 AQS 又依赖于 volatile 和 CAS(Compare-And-Swap)操作来实现线程的同步控制。其中AQS基本原理可见文章从ReentrantLock理解AQS的原理及应用总结,我们这里暂时增加一张原文图片进行体会理解:

现在我们来看一下几个关键方法:

lock()

获取锁。如果锁已经被其他线程持有,则当前线程将被阻塞,直到获取到锁。和 synchronized 没什么区别,如果获取不到锁,都会被阻塞;

lock.lock();
try {// critical section
} finally {lock.unlock();
}
unlock()

释放锁。通常在 finally 块中调用,以确保锁在使用之后总是被释放。重申需要手动加锁,然后在 finally 中解锁,在 finally 中解锁,在 finally 中解锁。

tryLock()

尝试获取锁。如果锁可用,则获取锁并返回 true,否则返回 false。该方法不会阻塞线程。

if (lock.tryLock()) {try {// critical section} finally {lock.unlock();}
} else {// handle lock not acquired
}
tryLock(long timeout, TimeUnit unit)

尝试在给定的时间范围内获取锁。如果在指定时间内获取到锁,则返回 true,否则返回 false

if (lock.tryLock(1, TimeUnit.SECONDS)) {try {// critical section} finally {lock.unlock();}
} else {// handle lock not acquired
}
lockInterruptibly()

获取锁,但与 lock() 不同的是,这个方法允许响应中断。如果线程在等待锁的过程中被中断,则抛出 InterruptedException

try {lock.lockInterruptibly();try {// critical section} finally {lock.unlock();}
} catch (InterruptedException e) {// handle interrupt
}

平时开发中建议在需要及时响应的业务场景下使用带超时时间的 tryLock 方法。基本建议如下:

  • 普通场景:使用 lock() 方法即可,确保线程安全。
  • 高并发、及时响应场景:使用带超时时间的 tryLock 方法,保证服务的高可用性和快速响应能力。

3. 读写锁ReentrantReadWriteLock

在高并发场景下,对于一些业务来说,使用 Lock 这种粗粒度的锁可能会导致性能瓶颈。例如,对于一个 HashMap,如果业务场景是读多写少,给读操作加上和写操作一样的锁会大大降低效率。因为在这种情况下,读操作会频繁发生,而每次读操作都被迫等待写锁的释放,这样就大大降低了系统的吞吐量。

为了解决这类问题,我们可以使用 ReentrantReadWriteLock(一种读写分离的锁机制)。ReentrantReadWriteLock 提供了两种锁:读锁(ReadLock)和写锁(WriteLock),其核心思想是将读操作和写操作分离开来,从而提高系统的并发性能。

基本内容说明
  • 读锁:允许多个线程同时获取读锁,进行并发读操作。读锁之间是共享的,多个读线程可以同时读取而不会相互阻塞。
  • 写锁:只有一个线程可以获取写锁,进行写操作。写锁之间是互斥的,写线程会阻塞其他写线程。
  • 读写互斥:读操作和写操作之间是互斥的,当一个线程持有写锁时,其他线程不能获取读锁。这样保证了数据的一致性和线程安全。
性能验证说明

为了更直观地对比 ReentrantReadWriteLock 在读多写少场景中的性能优势,我们可以编写一个性能测试用例,比较使用 ReentrantReadWriteLockReentrantLock 的性能差异。

private static final int NUM_OPERATIONS = 10000;
private static final int NUM_THREADS = 10;public static void main(String[] args) throws InterruptedException {// Test with ReentrantLocklong startTime = System.currentTimeMillis();Thread[] threads = new Thread[NUM_THREADS];for (int i = 0; i < NUM_THREADS; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS / NUM_THREADS; j++) {reentrantLock.lock();try {map.put("key" + j, "value" + j);} finally {reentrantLock.unlock();}}});}for (Thread thread : threads) {thread.start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("ReentrantLock Write Time: " + (endTime - startTime) + " ms");// Test with ReentrantReadWriteLockstartTime = System.currentTimeMillis();for (int i = 0; i < NUM_THREADS; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS / NUM_THREADS; j++) {writeLock.lock();try {map.put("key" + j, "value" + j);} finally {writeLock.unlock();}}});}for (Thread thread : threads) {thread.start();}for (Thread thread : threads) {thread.join();}endTime = System.currentTimeMillis();System.out.println("ReentrantReadWriteLock Write Time: " + (endTime - startTime) + " ms");
}

运行结果如下:

  • ReentrantLock 的写入时间为 63 ms;
  • ReentrantReadWriteLock 的写入时间为 32 ms

这个结果看起来ReentrantReadWriteLock 的写入时间比 ReentrantLock 要快,这表明在这种情况下,读写分离的锁确实能够提高性能。

4.乐观读取、悲观读取和写入的机制:StampedLock

StampedLock 是在 Java 8 中引入的,它提供了一种乐观读取、悲观读取和写入的机制,可以比 ReentrantReadWriteLock 更高效地支持读写分离。StampedLock 不支持重入,因此在使用时需要特别注意避免死锁的情况。

基本内容说明

StampedLock 主要有三种锁模式:

  • 写锁(writeLock):与普通的互斥锁类似,一次只能被一个线程持有。当一个线程持有写锁时,任何其他线程试图获取写锁或者悲观读锁都会被阻塞。
  • 悲观读锁(readLock):与 ReentrantReadWriteLock 中的读锁类似,用于读取共享数据。当一个线程持有悲观读锁时,其他线程试图获取写锁的请求会被阻塞,但不会阻塞其他悲观读锁的获取请求。
  • 乐观读锁(tryOptimisticRead):是一种乐观的读取模式,允许多个线程同时访问共享数据,不会阻塞其他线程的写入操作。但在使用乐观读锁时,需要通过 validate 方法来验证读取操作是否有效。

通过合理地选择锁模式,可以使 StampedLock 在某些情况下比传统的读写锁更高效。

性能验证说明

使用 StampedLockReentrantReadWriteLock 来实现读写分离,并比较它们的性能:

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.concurrent.locks.StampedLock;/*** @program: zyfboot-javabasic* @description: 使用 StampedLock 和 ReentrantReadWriteLock 来实现读写分离,并比较它们的性能。* @author: zhangyanfeng* @create: 2024-06-07 22:35**/
public class StampedLockVsReentrantReadWriteLock {private static final int NUM_THREADS = 200;private static final int NUM_OPERATIONS = 5000000;private static volatile int sharedVariable = 0;public static void main(String[] args) throws InterruptedException {long start, end;// Test with StampedLockStampedLock stampedLock = new StampedLock();start = System.currentTimeMillis();Thread[] stampedLockWriteThreads = new Thread[NUM_THREADS];Thread[] stampedLockReadThreads = new Thread[NUM_THREADS];for (int i = 0; i < NUM_THREADS; i++) {stampedLockWriteThreads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS; j++) {long stamp = stampedLock.writeLock();try {sharedVariable++;} finally {stampedLock.unlockWrite(stamp);}}});stampedLockWriteThreads[i].start();stampedLockReadThreads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS; j++) {long stamp = stampedLock.readLock();try {int value = sharedVariable;} finally {stampedLock.unlockRead(stamp);}}});stampedLockReadThreads[i].start();}for (Thread thread : stampedLockWriteThreads) {thread.join();}for (Thread thread : stampedLockReadThreads) {thread.join();}end = System.currentTimeMillis();System.out.println("StampedLock Write and Read Time: " + (end - start) + " ms");// Test with ReentrantReadWriteLockReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();Lock writeLock = rwLock.writeLock();Lock readLock = rwLock.readLock();start = System.currentTimeMillis();Thread[] rwLockWriteThreads = new Thread[NUM_THREADS];Thread[] rwLockReadThreads = new Thread[NUM_THREADS];for (int i = 0; i < NUM_THREADS; i++) {rwLockWriteThreads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS; j++) {writeLock.lock();try {sharedVariable++;} finally {writeLock.unlock();}}});rwLockWriteThreads[i].start();rwLockReadThreads[i] = new Thread(() -> {for (int j = 0; j < NUM_OPERATIONS; j++) {readLock.lock();try {int value = sharedVariable;} finally {readLock.unlock();}}});rwLockReadThreads[i].start();}for (Thread thread : rwLockWriteThreads) {thread.join();}for (Thread thread : rwLockReadThreads) {thread.join();}end = System.currentTimeMillis();System.out.println("ReentrantReadWriteLock Write and Read Time: " + (end - start) + " ms");}
}

运行结果如下:

  • StampedLock 的写入和读取时间为 21200 ms;
  • ReentrantReadWriteLock 的写入和读取时间为 26779 ms

结果来看StampedLock 的性能优于 ReentrantReadWriteLock。StampedLock 在这种情况下的表现更好,这与其内部实现机制有关。StampedLock 使用乐观读锁来提高并发性,而 ReentrantReadWriteLock 使用悲观读锁。在适当的情况下,StampedLock 能够更高效地支持读写分离,这通常在读操作远远多于写操作的情况下更为明显。

5. 公平锁与非公平锁

非公平锁允许在释放锁时不考虑等待队列中的其他线程的情况,新的线程可以立即争抢锁。这种情况下,可能存在某些线程总是无法获取锁的情况,造成线程饥饿。

相反,公平锁确保等待时间最长的线程会被优先选择来获取锁,这样每个线程都有机会获取到锁,避免了饥饿现象。在公平锁中,线程按照请求锁的顺序进入队列,并按顺序获取锁。

synchronized关键字 vs Lock接口

在Java中,synchronized关键字使用的是非公平锁,而在Lock接口的实现类中,可以通过构造函数来选择使用公平锁或非公平锁。

public ReentrantReadWriteLock(boolean fair) {sync = fair ? new FairSync() : new NonfairSync();readerLock = new ReadLock(this);writerLock = new WriteLock(this);
}

公平锁的实现需要维护一个有序的等待队列,因此在多核场景下,可能会降低吞吐量。这个在一开始讲Lock的时候我们就已经提过了,这里图在放一下:

功能验证
package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 比较非公平锁和公平锁的性能* @author: zhangyanfeng* @create: 2024-06-08 00:07**/
public class LockPerformanceComparison {private static final int NUM_THREADS = 10;private static final int NUM_ITERATIONS = 1000000;private static final Lock unfairLock = new ReentrantLock();private static final Lock fairLock = new ReentrantLock(true); // 公平锁public static void main(String[] args) {System.out.println("Unfair Lock Performance Test");testLock(unfairLock);System.out.println("Fair Lock Performance Test");testLock(fairLock);}private static void testLock(Lock lock) {long startTime = System.currentTimeMillis();Thread[] threads = new Thread[NUM_THREADS];for (int i = 0; i < NUM_THREADS; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < NUM_ITERATIONS; j++) {lock.lock();try {// 模拟一些计算或任务Math.pow(Math.random(), Math.random());} finally {lock.unlock();}}});}for (Thread thread : threads) {thread.start();}for (Thread thread : threads) {try {thread.join();} catch (InterruptedException e) {e.printStackTrace();}}long endTime = System.currentTimeMillis();System.out.println("Execution Time: " + (endTime - startTime) + " ms");}
}

运行结果如下:

  • Unfair Lock Performance Test:Execution Time: 3338 ms
  • Fair Lock Performance Test:Execution Time: 26218 ms

可以看到非公平锁(Unfair Lock)的性能明显优于公平锁(Fair Lock)。这是因为公平锁需要维护一个有序的等待队列,以确保线程按照它们的请求顺序获得锁。相比之下,非公平锁允许线程在锁可用时立即获取锁,而不考虑它们的请求顺序,因此效率更高。

在实际应用中,如果不需要严格的线程调度顺序,通常会选择使用非公平锁来获得更好的性能。

(三)Java 中两种加锁方式对比和建议

类别SynchronizedLock
实现方式monitorAQS
底层细节JVM优化Java API
分级锁
功能特性单一丰富
锁分离读写锁
锁超时带超时时间的 tryLock
可中断lockInterruptibly

Lock 的功能是比 Synchronized 多的,能够对线程行为进行更细粒度的控制。但如果只是用最简单的锁互斥功能,建议直接使用 Synchronized,有两个原因:

  • Synchronized 的编程模型更加简单,更易于使用
  • Synchronized 引入了偏向锁,轻量级锁等功能,能够从 JVM 层进行优化,同时JIT 编译器也会对它执行一些锁消除动作。

三、锁的优化手段

Java 中有两种加锁的方式:一种就是常见的synchronized 关键字,另外一种,就是使用 concurrent 包里面的 Lock。针对这两种锁,JDK 自身做了很多的优化,它们的实现方式也是不同的。这个在上文中已经讲了很多,我们体会的已经比较深了,现在站在巨人的肩膀上总结一下优化的一般思路:减少锁的粒度、减少锁持有的时间、锁分级、锁分离 、锁消除、乐观锁、无锁等。

(一)减少锁的粒度

锁粒度(Lock Granularity)指的是锁定的资源范围大小。锁的粒度越大,意味着锁定的资源范围越广,可能会导致更多的线程阻塞等待锁的释放;锁的粒度越小,意味着锁定的资源范围越窄,可以减少线程的阻塞和等待时间。

减少锁粒度是指通过细化锁的范围和控制的资源,来减少线程之间的冲突和竞争,从而提高并发性和系统性能。

假设我们有一个大型整数数组,多个线程需要同时对该数组进行读写操作。我们可以将数组分成多个段,每个段使用一个独立的锁,从而允许不同线程并行访问不同段的数据。

先看使用单个锁的 SingleLockArray:使用一个全局锁来保护整个数组。所有线程在访问数组时都需要获取这把锁,因此会导致更多的锁竞争和阻塞。

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 使用一个全局锁来保护整个数组。所有线程在访问数组时都需要获取这把锁,因此会导致更多的锁竞争和阻塞。* @author: zhangyanfeng* @create: 2024-06-08 01:02**/
public class SingleLockArray {private static final int ARRAY_SIZE = 10000;private final int[] array = new int[ARRAY_SIZE];private final Lock lock = new ReentrantLock();public void increment(int index) {lock.lock();try {array[index]++;} finally {lock.unlock();}}public int get(int index) {lock.lock();try {return array[index];} finally {lock.unlock();}}
}

在看使用分段锁的 SegmentLockArray:将数组分成多个段,每个段使用一个独立的锁。这样,当多个线程访问不同段的数据时,可以并行执行,而不会相互阻塞。

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 将数组分成多个段,每个段使用一个独立的锁。这样,当多个线程访问不同段的数据时,可以并行执行,而不会相互阻塞* @author: zhangyanfeng* @create: 2024-06-08 01:06**/
public class SegmentLockArray {private static final int NUM_SEGMENTS = 10;private static final int SEGMENT_SIZE = 1000;private static final int ARRAY_SIZE = NUM_SEGMENTS * SEGMENT_SIZE;private final int[] array = new int[ARRAY_SIZE];private final Lock[] locks = new ReentrantLock[NUM_SEGMENTS];public SegmentLockArray() {for (int i = 0; i < NUM_SEGMENTS; i++) {locks[i] = new ReentrantLock();}}public void increment(int index) {int segment = index / SEGMENT_SIZE;locks[segment].lock();try {array[index]++;} finally {locks[segment].unlock();}}public int get(int index) {int segment = index / SEGMENT_SIZE;locks[segment].lock();try {return array[index];} finally {locks[segment].unlock();}}}

进行验证说明

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.ThreadLocalRandom;/*** @program: zyfboot-javabasic* @description: 对比* @author: zhangyanfeng* @create: 2024-06-08 01:07**/
public class LockArrayPerComparison {private static final int NUM_SEGMENTS = 10;private static final int SEGMENT_SIZE = 1000;private static final int ARRAY_SIZE = NUM_SEGMENTS * SEGMENT_SIZE;public static void main(String[] args) throws InterruptedException {SegmentLockArray array = new SegmentLockArray();int numThreads = 100;Thread[] threads = new Thread[numThreads];// Writing threadsfor (int i = 0; i < numThreads / 2; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < 10000; j++) {int index = ThreadLocalRandom.current().nextInt(ARRAY_SIZE);array.increment(index);}});}// Reading threadsfor (int i = numThreads / 2; i < numThreads; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < 10000; j++) {int index = ThreadLocalRandom.current().nextInt(ARRAY_SIZE);array.get(index);}});}long startTime = System.currentTimeMillis();for (Thread thread : threads) {thread.start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Execution Time with Segment Locks: " + (endTime - startTime) + " ms");// Compare with single lockSingleLockArray singleLockArray = new SingleLockArray();Thread[] singleLockThreads = new Thread[numThreads];// Writing threadsfor (int i = 0; i < numThreads / 2; i++) {singleLockThreads[i] = new Thread(() -> {for (int j = 0; j < 10000; j++) {int index = ThreadLocalRandom.current().nextInt(ARRAY_SIZE);singleLockArray.increment(index);}});}// Reading threadsfor (int i = numThreads / 2; i < numThreads; i++) {singleLockThreads[i] = new Thread(() -> {for (int j = 0; j < 10000; j++) {int index = ThreadLocalRandom.current().nextInt(ARRAY_SIZE);singleLockArray.get(index);}});}startTime = System.currentTimeMillis();for (Thread thread : singleLockThreads) {thread.start();}for (Thread thread : singleLockThreads) {thread.join();}endTime = System.currentTimeMillis();System.out.println("Execution Time with Single Lock: " + (endTime - startTime) + " ms");}
}

运行结果如下:

  • Execution Time with Segment Locks: 94 ms
  • Execution Time with Single Lock: 40 ms

可以看到使用分段锁的 SegmentLockArray 的执行时间明显优于使用单个锁的 SingleLockArray,特别是在高并发的场景下。这样,通过减少锁粒度,我们可以显著提高系统的并发性能。

(二)减少锁持有时间

通过让锁资源尽快地释放,减少锁持有的时间,其他线程可更迅速地获取锁资源,进行其他业务的处理。

我们使用两个计数器类,一个是未优化的版本,另一个是优化后的版本。

未优化的版本(UnoptimizedCounter):具体代码如下展示,在锁定的代码块中进行了模拟的工作(Thread.sleep(1)),导致锁的持有时间较长,从而增加了锁竞争和线程阻塞的时间。

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 未优化的版本* @author: zhangyanfeng* @create: 2024-06-08 01:22**/
public class UnoptimizedCounter {private int count = 0;private final Lock lock = new ReentrantLock();public void increment() {lock.lock();try {// Simulate some work that doesn't need to be lockedThread.sleep(1);count++;} catch (InterruptedException e) {e.printStackTrace();} finally {lock.unlock();}}public int getCount() {lock.lock();try {return count;} finally {lock.unlock();}}
}

优化后的版本(OptimizedCounter):将模拟的工作(`Thread.sleep(1))移到了锁定代码块之外。锁的持有时间大幅减少,锁定代码块仅包含了必要的操作(count++),减少了锁竞争和线程阻塞的时间,从而提高了系统的并发性能。

package org.zyf.javabasic.thread.lock.opti;import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;/*** @program: zyfboot-javabasic* @description: 优化后的版本* @author: zhangyanfeng* @create: 2024-06-08 01:23**/
public class OptimizedCounter {private int count = 0;private final Lock lock = new ReentrantLock();public void increment() {try {// Simulate some work that doesn't need to be lockedThread.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}lock.lock();try {count++;} finally {lock.unlock();}}public int getCount() {lock.lock();try {return count;} finally {lock.unlock();}}
}

我们将通过创建多个线程对这两个计数器进行并发操作,并测量它们的执行时间:

package org.zyf.javabasic.thread.lock.opti;/*** @program: zyfboot-javabasic* @description: LockPerformanceTest* @author: zhangyanfeng* @create: 2024-06-08 01:24**/
public class LockTimePerformanceTest {private static final int NUM_THREADS = 10;private static final int NUM_ITERATIONS = 1000;public static void main(String[] args) throws InterruptedException {System.out.println("UnoptimizedCounter Performance Test");UnoptimizedCounter unoptimizedCounter = new UnoptimizedCounter();testCounterPerformance(unoptimizedCounter);System.out.println("OptimizedCounter Performance Test");OptimizedCounter optimizedCounter = new OptimizedCounter();testCounterPerformance(optimizedCounter);}private static void testCounterPerformance(Object counter) throws InterruptedException {Thread[] threads = new Thread[NUM_THREADS];long startTime = System.currentTimeMillis();for (int i = 0; i < NUM_THREADS; i++) {threads[i] = new Thread(() -> {for (int j = 0; j < NUM_ITERATIONS; j++) {if (counter instanceof UnoptimizedCounter) {((UnoptimizedCounter) counter).increment();} else if (counter instanceof OptimizedCounter) {((OptimizedCounter) counter).increment();}}});threads[i].start();}for (Thread thread : threads) {thread.join();}long endTime = System.currentTimeMillis();System.out.println("Execution Time: " + (endTime - startTime) + " ms");if (counter instanceof UnoptimizedCounter) {System.out.println("Final count (Unoptimized): " + ((UnoptimizedCounter) counter).getCount());} else if (counter instanceof OptimizedCounter) {System.out.println("Final count (Optimized): " + ((OptimizedCounter) counter).getCount());}}
}

运行结果如下:

  • UnoptimizedCounter Performance Test:Execution Time: 12656 ms
  • OptimizedCounter Performance Test:Execution Time: 1254 ms

优化后的版本(OptimizedCounter)应该会显示更短的执行时间,从而验证了减少锁持有时间对性能的优化效果。

(三)锁分级

JVM中的锁分级机制主要包括偏向锁、轻量级锁和重量级锁。这些锁的分级和升级是为了在不同的线程竞争情况下,尽可能地减少锁的开销和提升性能。

锁的升级过程是不可逆的,即从偏向锁到轻量级锁,再到重量级锁。如果一个锁已经升级为重量级锁,即使之后的竞争减少,也不会降级回轻量级锁或偏向锁。这样做是为了简化JVM的实现,避免复杂的锁降级逻辑带来的额外开销。

这个在上文第二部分的开头就已经讲过了。

(四)锁分离

锁分离是一种针对不同类型操作进行区分的优化技术,通过使用不同的锁来分别处理读操作和写操作,来提高并发性能。读写锁(ReentrantReadWriteLock)就是一种典型的锁分离技术的实现。锁分离的核心思想在于,读操作和写操作对资源的影响不同,可以采用不同的锁策略来优化性能。

其优化版本StampedLock 的性能优于 ReentrantReadWriteLock,StampedLock 使用乐观读锁来提高并发性,而 ReentrantReadWriteLock 使用悲观读锁。在适当的情况下,StampedLock 能够更高效地支持读写分离,这通常在读操作远远多于写操作的情况下更为明显。

具体验证代码在上方第二部分的Lock中已经给出了样例,请使用中自行选择。

(五)锁消除

锁消除(Lock Elimination)是指 JVM 通过分析代码的运行范围,判断出某些锁在多线程环境下没有竞争,因此可以去掉这些锁操作。这个过程是由 JVM 在运行时通过即时编译器(JIT 编译器)和逃逸分析来决定的。

逃逸分析(Escape Analysis)是锁消除的关键技术。它分析对象的作用范围,判断对象是否会逃逸出方法或线程。如果某个对象只在方法内部使用,并且不会逃逸出这个方法,则认为这个对象是线程私有的,锁操作就没有实际意义,可以消除。

考虑以下两个字符串拼接的示例:

package org.zyf.javabasic.thread.lock.opti;/*** @program: zyfboot-javabasic* @description: 两个字符串拼接的示例* @author: zhangyanfeng* @create: 2024-06-08 01:49**/
public class LockEliminationExample {public static void main(String[] args) {long startTime;long endTime;// Test with StringBufferstartTime = System.currentTimeMillis();for (int i = 0; i < 1000000; i++) {concatenateStringBuffer("Hello", "World");}endTime = System.currentTimeMillis();System.out.println("StringBuffer Execution Time: " + (endTime - startTime) + " ms");// Test with StringBuilderstartTime = System.currentTimeMillis();for (int i = 0; i < 1000000; i++) {concatenateStringBuilder("Hello", "World");}endTime = System.currentTimeMillis();System.out.println("StringBuilder Execution Time: " + (endTime - startTime) + " ms");}public static String concatenateStringBuffer(String s1, String s2) {StringBuffer sb = new StringBuffer();sb.append(s1);sb.append(s2);return sb.toString();}public static String concatenateStringBuilder(String s1, String s2) {StringBuilder sb = new StringBuilder();sb.append(s1);sb.append(s2);return sb.toString();}
}

StringBufferStringBuilder 都用于字符串拼接,但 StringBuffer 是线程安全的,通过内部使用的同步机制(锁)来保证线程安全,而 StringBuilder 则是非线程安全的,没有同步机制。

当 JVM 通过逃逸分析发现 StringBuffer 对象没有逃逸出方法时,就会将其锁操作消除,从而使得它的性能和 StringBuilder 接近。

(六)乐观锁

乐观锁认为自己在使用数据时不会有别的线程修改数据,所以不会添加锁,只是在更新数据的时候去判断之前有没有别的线程更新了这个数据。如果这个数据没有被更新,当前线程将自己修改的数据成功写入。如果数据已经被其他线程更新,则根据不同的实现方式执行不同的操作(例如报错或者自动重试)

在Java中是通过使用无锁编程来实现,最常采用的是CAS算法,Java原子类中的递增操作就通过CAS自旋实现的。具体的理论和使用验证可见:Java中常用的锁总结与理解

当然最直接的可见:超越并发瓶颈:CAS与乐观锁的智慧应用

(七)无锁

无锁队列是一种在多线程环境下访问共享资源的方式,它不会阻塞其他线程的执行。在 Java 中,最典型的无锁队列实现是 ConcurrentLinkedQueue,它使用CAS(Compare and Swap)指令来处理对数据的并发访问。CAS指令是一种非阻塞的原子操作,不会引起上下文切换和线程调度,因此是一种非常轻量级的多线程同步机制。

ConcurrentLinkedQueue的实现基于CAS机制,它将队列的入队和出队等操作拆分为更细粒度的步骤,进一步减小了CAS控制的范围,提高了并发性能。与之相对应的是阻塞队列LinkedBlockingQueue,它内部使用锁机制来实现同步,因此性能没有无锁队列高。

除了ConcurrentLinkedQueue外,还有一种无锁队列框架叫做Disruptor,它是一个无锁、有界的队列框架,具有极高的性能。Disruptor使用RingBuffer、无锁和缓存行填充等技术来实现高效的并发访问,适用于极高并发的场景,比如日志、消息等中间件。尽管Disruptor的编程模型相对复杂,但在需要极致性能的场景下,可以取代传统的阻塞队列。

参考文章

Java synchronized 关键字(3)-JVM 重量级锁 Monitor 的实现_jvm中monitor的实现原理-CSDN博客

JAVA多线程学习-------monitor所实现原理_java objectmonitor 原理-CSDN博客

Monitor工作原理&synchronized锁膨胀过程及其优化_rom monitor软件的工作原理-CSDN博客

synchronized底层monitor原理_synchronized monitor原理-CSDN博客

https://zhuanlan.zhihu.com/p/581418806

https://juejin.cn/post/7131739806156980232

https://www.51cto.com/article/743092.html

浅谈一下Java中的几种JVM级别的锁_java_脚本之家

聊聊 Java 的几把 JVM 级锁_jvm层次的锁都有哪些-CSDN博客

https://www.cnblogs.com/wzj4858/p/8215369.html

Java基础-2、并发_分段锁的原理,锁力度减小的思考-CSDN博客

JAVA锁机制-可重入锁,可中断锁,公平锁,读写锁,自旋锁_可中断锁实现原理-CSDN博客

Java中的线程和锁机制_第三 提供的接 审核threadlocal,synchronized等锁机制以及线程池等技术的底-CSDN博客

ReentrantReadWriteLock读写锁-CSDN博客

读写锁ReentrantReadWriteLock&StampLock详解_reentrantreadwritelock和stampedlock-CSDN博客

https://www.cnblogs.com/xiaoxi/p/9140541.html

ReentrantReadWriteLock.ReadLock (Java 2 Platform SE 5.0)

深入理解读写锁ReentrantReadWriteLock_读写锁会产生什么问题-CSDN博客

Java并发编程学习篇3_读写锁ReadWriteLock、阻塞队列BlockingQueue、同步队列SynchronousQueue、线程池(三大方法、七大参数、四种拒绝策略、原生方式创建线程池)_reentrantreadwrihtlock有队列吗-CSDN博客

StampedLock (Java Platform SE 8 )

StampedLock Class (Java.Util.Concurrent.Locks) | Microsoft Learn

https://www.cnblogs.com/myworld7/p/12332911.html

StampedLock (Java SE 19 & JDK 19 [build 1])

https://www.cnblogs.com/zhujiqian/p/16898222.html

开放平台

一文搞懂ReentrantLock的公平锁和非公平锁_ReentrantLock_Ayue、_InfoQ写作社区

Java多线程 -- 公平锁和非公平锁的一些思考-阿里云开发者社区

码农会锁,ReentrantLock之公平锁讲解和实现 | HeapDump性能社区

滑动验证页面

深入剖析 ReentrantLock 公平锁与非公平锁源码实现 - AIQ

13 案例分析:多线程锁的优化

https://blog.51cto.com/panyujie/5478573

Synchronize锁优化手段有哪些_牛客博客

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器学习面试-常见题目

文章目录 一、框架问题1. 监督学习和无监督学习有什么不同&#xff1f;2. 什么是深度学习&#xff0c;它与机器学习算法之间有什么联系&#xff1f;3. 如何评估机器学习模型的有效性&#xff1f;4. 如何确保模型没有过拟合&#xff1f;5. 什么是核技巧&#xff0c;有什么用处&a…

4_机械臂坐标系简介

一、坐标系的标准命名 为了规范起见&#xff0c;有必要给机器人和工作空间专门命名和确定专门的“标准”坐标系。 图3-27为一种典型的工况&#xff0c;机器人抓持某种工具&#xff0c;并把工具末端移动到操作者指定的位置。图3-27所示的5个坐标系就是需要命名的坐标系。这五个坐…

7z及7zip-cpp最高压缩比的免费开源压缩软件

7z介绍 7z是一种主流高效的压缩格式&#xff0c;它拥有极高的压缩比。在计算机科学中&#xff0c;7z是一种可以使用多种压缩算法进行数据压缩的档案格式。该格式最初由7-Zip实现并采用&#xff0c;但这种档案格式是公有的&#xff0c;并且7-Zip软件本身亦在GNU宽通用公共许可证…

PCIe总线-RK3588 PCIe子系统简介(八)

1.PCIe子系统 RK3588 PCIe子系统如下图所示。总共拥有5个PCIe控制器。PCIe30X4(4L)支持RC和EP模式&#xff0c;其他4个仅支持RC模式。ITS port 1连接PCIe30X4(4L)和PCIe30X2(2L)控制器&#xff0c;PCIe30X4(4L)和PCIe30X2(2L)控制器使用PCIe3.0 PIPE PHY。ITS port 0连接PCIe3…

RIP路由附加度量值(华为)

#交换设备 RIP路由附加度量值 RIP&#xff08;Routing Information Protocol&#xff09;路由协议中的附加度量值是指在RIP路由原来度量值的基础上所增加的额外度量值&#xff0c;通常以跳数来表示。这个附加度量值可以是正值&#xff0c;也可以是负值&#xff0c;用于影响路…

关于STM32上用HID HOST调鼠标数据的解析

一、前言 关于这章主要是基于我前面的那篇文章 链接: 关于怎么用Cubemx生成的USBHID设备实现读取一体的鼠标键盘设备&#xff08;改进版&#xff09; https://blog.csdn.net/qq_29187987/article/details/139535648?spm1001.2014.3001.5501 引用的文章的简介 引用的这篇文…

数据库概述1

数据&#xff1a;描述事物的符号记录称为数据&#xff1b; 包括数字、图片、音频等&#xff1b; 数据库&#xff1a;长期储存在计算机内有组织、可共享的大量数据的集合&#xff1b;数据库中的数据按照一定的数据模型组织、描述和存储&#xff0c;具有较小的数据冗余、较高的数…

STM32学习笔记(一)--时钟树详解

&#xff08;1&#xff09;时钟概述&#xff1b;时钟是具有周期性的脉冲信号&#xff0c;最常用的是占空比50%的方波。&#xff08;时钟相当于单片机的脉搏&#xff1b;STM32本身非常复杂&#xff0c;外设非常的多&#xff0c;为了保持低功耗工作&#xff0c;STM32 的主控默认不…

【Python】深入了解 AdaBoost:自适应提升算法

我们都找到天使了 说好了 心事不能偷藏着 什么都 一起做 幸福得 没话说 把坏脾气变成了好沟通 我们都找到天使了 约好了 负责对方的快乐 阳光下 的山坡 你素描 的以后 怎么抄袭我脑袋 想的 &#x1f3b5; 薛凯琪《找到天使了》 在机器学习的领域中&#x…

算法工程师 | 如何快速 了解,掌握一个算法!脚踏实地,迎着星辰,向前出发 ~

本文是一些碎碎念 希望对正在迈向 算法工程师道路的你 有所裨益 一般来说&#xff0c;代码 中会有很多 算法实现的细节&#xff0c;但论文可能并没有体现&#xff0c;所以能够尝试自己 仔细阅读论文&#xff0c;手动复现代码&#xff0c;基本上来说对 这个 算法 你有了全…

夏季城市环境卫生挑战多:TSINGSEE青犀智慧环卫方案助力城市垃圾站智能管理

一、背景分析 夏季&#xff0c;随着气温的攀升&#xff0c;城市垃圾的数量和种类也随之增加&#xff0c;这给环卫工作带来了极大的挑战。环卫垃圾站点作为城市垃圾处理的重要一环&#xff0c;其管理效率直接关系到城市环境的整洁与卫生。近年来&#xff0c;随着视频监控技术的…

【Redis】Redis常见问题——缓存更新/内存淘汰机制/缓存一致性

目录 回顾数据库的问题如何提高 mysql 能承担的并发量&#xff1f;缓存解决方案应对的场景 缓存更新问题定期生成如何定期统计定期生成的优缺点 实时生成maxmemory 设置成多少合适呢&#xff1f;项目类型上来说 新的问题 内存淘汰策略Redis淘汰策略为什么redis要内存淘汰内存淘…

ESP32 IDF ADF 加入音频

需要把mp3制作成音频bin 用ADF自带工具 果用户需要生成自己的 audio-esp.bin&#xff0c;则需要执行 mk_audio_bin.py 脚本&#xff08;位于 $ADF_PATH/tools/audio_tone/mk_audio_tone.py&#xff09;&#xff0c;并且指定相关文件的路径。 源 MP3 文件在 tone_mp3_folder …

红黑树(C++)

文章目录 写在前面1. 红黑树的概念及性质1. 1 红黑树的概念1. 2 红黑树的性质 2. 红黑树节点的定义3. 红黑树的插入3.1 按照二叉搜索的树规则插入新节点3.2 检测新节点插入后&#xff0c;红黑树的性质是否造到破坏 4.红黑树的删除5.红黑树的验证6.源码 写在前面 在上篇文章中&…

5.3.1_2 二叉树的层次遍历

&#x1f44b; Hi, I’m Beast Cheng&#x1f440; I’m interested in photography, hiking, landscape…&#x1f331; I’m currently learning python, javascript, kotlin…&#x1f4eb; How to reach me --> 458290771qq.com 喜欢《数据结构》部分笔记的小伙伴可以订…

c++模板模式

文章目录 模板模式什么是模板模式为什么使用模板模式模板模式实现步骤 示例模板模式优缺点 模板模式 什么是模板模式 模板模式&#xff08;Template Method Pattern&#xff09;是一种行为设计模式&#xff0c;它定义了一个操作中的算法骨架&#xff0c;将某些步骤的具体实现延…

[DDR4] DDR 简史

依公知及经验整理&#xff0c;原创保护&#xff0c;禁止转载。 专栏 《深入理解DDR4》 存和硬盘&#xff0c;这对电脑的左膀右臂&#xff0c;共同扛起了存储的重任。内存以其超凡的存取速度闻名&#xff0c;但一旦断电&#xff0c;内存中的数据也会消失。它就像我们的工作桌面&…

tokenization(二)子词切分方法

文章目录 概述BPE构建词表词元化代码实现 WordPieceUnigram估算概率&#xff08;E&#xff09;删除词元&#xff08;M&#xff09; 参考资料 概述 接上回&#xff0c;子词词元化&#xff08;Subwords tokenization&#xff09;是平衡字符级别和词级别的一种方法&#xff0c;也…

网络通信架构

BS架构/CS架构 使用协议分别对应&#xff1a; TCP / HTTP 在计算机网络和软件开发中&#xff0c;CS架构&#xff08;Client-Server Architecture&#xff0c;客户端-服务器架构&#xff09;和BS架构&#xff08;Browser-Server Architecture&#xff0c;浏览器-服务器架构&am…