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

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

相关文章

使用Python自建轻量级的HTTP调试工具

《使用Python自建轻量级的HTTP调试工具》这篇文章主要为大家详细介绍了如何使用Python自建一个轻量级的HTTP调试工具,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录一、为什么需要自建工具二、核心功能设计三、技术选型四、分步实现五、进阶优化技巧六、使用示例七、性能对比八、扩展方向建

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

java中反射(Reflection)机制举例详解

《java中反射(Reflection)机制举例详解》Java中的反射机制是指Java程序在运行期间可以获取到一个对象的全部信息,:本文主要介绍java中反射(Reflection)机制的相关资料... 目录一、什么是反射?二、反射的用途三、获取Class对象四、Class类型的对象使用场景1五、Class

MySQL中慢SQL优化的不同方式介绍

《MySQL中慢SQL优化的不同方式介绍》慢SQL的优化,主要从两个方面考虑,SQL语句本身的优化,以及数据库设计的优化,下面小编就来给大家介绍一下有哪些方式可以优化慢SQL吧... 目录避免不必要的列分页优化索引优化JOIN 的优化排序优化UNION 优化慢 SQL 的优化,主要从两个方面考虑,SQL 语

MySQL中慢SQL优化方法的完整指南

《MySQL中慢SQL优化方法的完整指南》当数据库响应时间超过500ms时,系统将面临三大灾难链式反应,所以本文将为大家介绍一下MySQL中慢SQL优化的常用方法,有需要的小伙伴可以了解下... 目录一、慢SQL的致命影响二、精准定位问题SQL1. 启用慢查询日志2. 诊断黄金三件套三、六大核心优化方案方案