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

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

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

MySQL中的锁机制详解之全局锁,表级锁,行级锁

《MySQL中的锁机制详解之全局锁,表级锁,行级锁》MySQL锁机制通过全局、表级、行级锁控制并发,保障数据一致性与隔离性,全局锁适用于全库备份,表级锁适合读多写少场景,行级锁(InnoDB)实现高并... 目录一、锁机制基础:从并发问题到锁分类1.1 并发访问的三大问题1.2 锁的核心作用1.3 锁粒度分

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查

Redis的持久化之RDB和AOF机制详解

《Redis的持久化之RDB和AOF机制详解》:本文主要介绍Redis的持久化之RDB和AOF机制,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述RDB(Redis Database)核心原理触发方式手动触发自动触发AOF(Append-Only File)核

PostgreSQL中MVCC 机制的实现

《PostgreSQL中MVCC机制的实现》本文主要介绍了PostgreSQL中MVCC机制的实现,通过多版本数据存储、快照隔离和事务ID管理实现高并发读写,具有一定的参考价值,感兴趣的可以了解一下... 目录一 MVCC 基本原理python1.1 MVCC 核心概念1.2 与传统锁机制对比二 Postg

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

Redis过期删除机制与内存淘汰策略的解析指南

《Redis过期删除机制与内存淘汰策略的解析指南》在使用Redis构建缓存系统时,很多开发者只设置了EXPIRE但却忽略了背后Redis的过期删除机制与内存淘汰策略,下面小编就来和大家详细介绍一下... 目录1、简述2、Redis http://www.chinasem.cn的过期删除策略(Key Expir

Go语言中Recover机制的使用

《Go语言中Recover机制的使用》Go语言的recover机制通过defer函数捕获panic,实现异常恢复与程序稳定性,具有一定的参考价值,感兴趣的可以了解一下... 目录引言Recover 的基本概念基本代码示例简单的 Recover 示例嵌套函数中的 Recover项目场景中的应用Web 服务器中