synchronized锁的升级

synchronized 是 Java 中用于实现多线程同步的关键字,用于确保在多线程环境下共享资源的安全访问。在讨论synchronized锁升级过程时,主要是指从无锁状态到有锁状态的升级,以及从偏向锁或轻量级锁升级到重量级锁的过程。

无锁状态到有锁状态的升级:

  • 偏向锁(Bias Locking): 在无锁状态下,当第一个线程访问同步代码块时,JVM会将对象头中的标志位设置为偏向锁,并将线程ID记录在对象头中。此后,当同一个线程再次访问该同步代码块时,无需加锁,因为偏向锁认为同一个线程会反复获取锁。这加速了单线程访问同步代码块的速度。
  • 轻量级锁(Lightweight Locking): 当其他线程尝试获取同一个锁时,偏向锁会升级为轻量级锁。JVM会尝试通过CAS(Compare-and-Swap)操作来获取锁,如果CAS成功,当前线程就获得了轻量级锁。如果失败,表示有其他线程在竞争锁,那么当前线程会进行自旋等待,而不是阻塞。

轻量级锁到重量级锁的升级:

  • 自旋锁(Spin Lock): 在轻量级锁阶段,如果自旋等待的次数超过一定阈值,或者发生竞争失败,JVM会将轻量级锁升级为自旋锁。此时,线程会进行自旋等待一段时间,希望其他线程会释放锁,而不是进入阻塞状态。
  • 重量级锁(Heavyweight Locking): 如果自旋锁的自旋等待时间过长或者自旋等待不成功,JVM会将锁升级为重量级锁。重量级锁会导致请求锁的线程进入阻塞状态,并且会导致操作系统级别的线程切换,性能开销相对较大。
    synchronized锁的升级_第1张图片

锁升级优点:

  • 性能优化: 锁升级可以根据不同的线程竞争情况,选择不同的锁级别,从而优化性能。在竞争不激烈的情况下,使用轻量级锁和自旋锁可以避免线程阻塞,提高效率。而在竞争激烈的情况下,升级到重量级锁可以避免过多的自旋等待,减少CPU消耗。
  • 减少线程切换: 轻量级锁和自旋锁的使用可以减少线程的阻塞和唤醒操作,降低了操作系统级别的线程切换开销,提高了程序的并发性能。 快速路径: 锁升级使得在无竞争的情况下,线程可以快速地获取锁,而无需进行复杂的同步操作,减少了开销。

锁升级缺点:

  • 内存开销: 锁升级过程中需要在对象头中存储额外的信息,如线程ID等,以及在某些情况下需要为自旋等待提供空间。这会增加每个对象的内存开销。
  • 自旋开销: 自旋锁可能会导致线程不断地忙等待,占用CPU资源。如果自旋时间过长或自旋不成功,会浪费大量的CPU时间。
  • 不确定性: 锁升级的机制可能因为不同的JVM实现和硬件平台而有所不同,可能会导致在某些情况下表现不一致,需要开发人员深入理解锁升级的原理。
  • 竞争情况: 在高竞争的情况下,使用自旋锁可能会导致多个线程同时自旋,进而增加了CPU的使用率,降低了性能。

你可能感兴趣的:(java,并发,java)