锁的优化机制(偏向锁、自旋锁、轻量级锁、重量级锁)

2024-01-30 13:20

本文主要是介绍锁的优化机制(偏向锁、自旋锁、轻量级锁、重量级锁),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

锁的状态从低到高依次为无锁->偏向锁->轻量级锁->重量级锁,升级的过程就是从低到高,降级在一定条件也是有可能发生的,优化机制包括自适应锁、自旋锁、锁消除、锁粗化、轻量级锁和偏向锁。
这边主要以synchronized、ReentrantLock两种实现方式来说明 偏向锁、自旋锁、轻量级锁、重量级锁

目录

  • 一、偏向锁
    • 基本概念
    • 基本实现
  • 二、自旋锁
    • 基本概念
    • 基本实现
  • 三、轻量级锁
    • 基本概念
    • 基本实现
  • 四、重量级锁
    • 基本概念
    • 基本实现
  • 五、锁粗化、锁消除
    • 锁粗化
    • 锁消除
  • 六、优缺点对比

一、偏向锁

基本概念

当线程访问同步块获取锁时,会在对象头和栈帧中的锁记录里存储偏向锁的线程ID,之后这个线程再次进入同步块时都不需要CAS来加锁和解锁了,偏向锁会永远偏向第一个获得锁的线程,如果后续没有其他线程获得过这个锁,持有锁的线程就永远不需要进行同步,反之,当有其他线程竞争偏向锁时,持有偏向锁的线程就会释放偏向锁。

基本实现

对于synchronized,偏向锁就用设置-XX:+UseBiasedLocking开启偏向锁

对于ReentrantLock,new ReentrantLock(false)(非公平锁)设置参数为false创建的是偏向锁

当偏向锁的获取出现竞争,则偏向锁可能会升级为轻量级锁,偏向锁适用于无竞争、竞争小的场景。

二、自旋锁

基本概念

自旋的概念主要就是一个忙等待,等待获取到锁后再进行下一步操作。

基本实现

对于synchronized,自旋锁可以通过设置-XX:+UseSpining来开启,自旋的默认次数是10次,可以使用-XX:PreBlockSpin设置。

对于ReentrantLock,通过逻辑设置忙等待处理,以下是示例。

import java.util.concurrent.locks.ReentrantLock;/*** 自旋锁*/
public class SpinLockExample {private static int count = 0;private static final ReentrantLock lock = new ReentrantLock();public static void main(String[] args) {Thread t1 = new Thread(() -> {for (int i = 0; i < 1000; i++) {try {while (!lock.tryLock()) {// 自旋等待获得锁}count++;} finally {lock.unlock();}}});Thread t2 = new Thread(() -> {for (int i = 0; i < 1000; i++) {try {while (!lock.tryLock()) {// 自旋等待获得锁}count++;} finally {lock.unlock();}}});t1.start();t2.start();try {t1.join();t2.join();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Count: " + count);}
}

三、轻量级锁

基本概念

在轻量级锁状态下继续锁竞争,如果成功就成功获取轻量级锁。否则进入锁膨胀阶段,没有抢到锁的线程将自旋,即不停地循环判断锁是否能够被成功获取。长时间的自旋操作是非常消耗资源的,一个线程持有锁,其他线程就只能在原地空耗CPU,执行不了任何有效的任务,这种现象叫做忙等(busy-waiting)。如果锁竞争情况严重,某个达到最大自旋次数的线程,会将轻量级锁升级为重量级锁。

基本实现

synchronized所修饰的方法所访问的对象是静态对象时,确实可以认为它是轻量级锁。因为静态对象在类加载时被加载到方法区,并且只会被加载一次。当多个线程访问静态方法时,由于它们访问的是同一个静态对象,所以实际上只有一个线程能够进入同步块,从而实现轻量级锁的效果。

ReentrantLock默认使用的是轻量级锁。但是,当锁被频繁地获取和释放,或者在锁竞争的情况下,ReentrantLock 会将轻量级锁升级为重量级锁,以避免锁竞争和提高性能。

四、重量级锁

基本概念

重量级锁即是需要排队竞争锁,当锁被占用时则需要挂起,等待锁释放后唤醒线程竞争获取锁。在锁资源被占用时会进行同步操作,以确保只有一个线程能够访问锁资源。

基本实现

当使用 synchronized 关键字时,如果锁对象是一个实例对象,那么就是重量级同步。这种同步方式会涉及到锁对象的实例化,需要将锁对象的状态存储在内存中,因此占用大量的系统资源,性能较低。(自动切换轻量级锁和重量级锁)

对于ReentrantLock,以下是示例。

import java.util.concurrent.locks.ReentrantLock;public class ReentrantLockExample {private final ReentrantLock lock = new ReentrantLock();private int counter = 0;public void incrementCounter() {lock.lock(); // 获取锁try {counter++;} finally {lock.unlock(); // 释放锁}}public int getCounter() {return counter;}
}

当多次调用 incrementCounter() 方法时,如果锁没有被其他线程占用,那么 ReentrantLock 会自动重入锁,不会导致死锁。但如果锁被其他线程占用,那么 ReentrantLock 会将轻量级锁升级为重量级锁,从而避免锁竞争,提高性能。

五、锁粗化、锁消除

锁粗化

锁粗化指的是有很多操作都是对同一个对象进行加锁,就会把锁的同步范围扩展到整个操作序列之外。(指在某些情况下,放宽对锁的使用限制,从而避免死锁的发生)
例如:可以ReentrantLock使用公平锁来保证等待锁释放的线程按照请求锁的顺序来释放锁,从而避免死锁的发生,同时也可以通过减少锁的使用时间、优化锁的同步策略等方法来避免死锁的发生。

锁消除

锁消除指的是JVM检测到一些同步的代码块,完全不存在数据竞争的场景,也就是不需要加锁,就会进行锁消除。

六、优缺点对比

借用下别的博主的总结
其他博主的总结借用

复习阶段可能个人理解不一定准确,有问题望大家指出,谢谢❀

这篇关于锁的优化机制(偏向锁、自旋锁、轻量级锁、重量级锁)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/660367

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

Java如何通过反射机制获取数据类对象的属性及方法

《Java如何通过反射机制获取数据类对象的属性及方法》文章介绍了如何使用Java反射机制获取类对象的所有属性及其对应的get、set方法,以及如何通过反射机制实现类对象的实例化,感兴趣的朋友跟随小编一... 目录一、通过反射机制获取类对象的所有属性以及相应的get、set方法1.遍历类对象的所有属性2.获取

MySQL中的锁和MVCC机制解读

《MySQL中的锁和MVCC机制解读》MySQL事务、锁和MVCC机制是确保数据库操作原子性、一致性和隔离性的关键,事务必须遵循ACID原则,锁的类型包括表级锁、行级锁和意向锁,MVCC通过非锁定读和... 目录mysql的锁和MVCC机制事务的概念与ACID特性锁的类型及其工作机制锁的粒度与性能影响多版本

MySQL不使用子查询的原因及优化案例

《MySQL不使用子查询的原因及优化案例》对于mysql,不推荐使用子查询,效率太差,执行子查询时,MYSQL需要创建临时表,查询完毕后再删除这些临时表,所以,子查询的速度会受到一定的影响,本文给大家... 目录不推荐使用子查询和JOIN的原因解决方案优化案例案例1:查询所有有库存的商品信息案例2:使用EX

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件