JavaEE:多线程进阶(常见的锁策略)

news2025/1/11 21:53:19

文章目录

  • 常见的锁策略
    • 各种锁的概念
  • synchronized
    • 特点
    • 加锁过程
  • 锁消除(编译器的优化策略)
  • 锁粗化(编译器的优化策略)

常见的锁策略

锁是一个非常广义的问题.
synchronized只是市面上五花八门的锁的一种典型的实现.它是Java内置的,推荐使用的锁.

各种锁的概念

下面这些概念,一般面试的时候,不会直接问你,但是可能会在某某问题中,引出这样的术语.

  1. 乐观锁 vs 悲观锁

    • 乐观锁: 加锁的时候,假设出现锁冲突的概率不大,接下来要围绕加锁做的工作,就会更少.
    • 悲观锁: 加锁的时候,假设出现锁冲突的概率很大,接下来围绕加锁要做的工作,就会更多

    synchronized这把锁算是自适应的.
    synchronized初始情况下是乐观的,同时它会在背后偷偷统计锁冲突了多少次.如果发现锁冲突的次数达到一定程度了,就会变为悲观的.

  2. 重量级锁 vs 轻量级锁

    • 重量级锁: 加锁的开销比较大,要做更多的工作.(往往悲观的时候,会做的重)
    • 轻量级锁: 加锁的开销比较小,要做的工作相对更少.(往往乐观的时候,会做的轻)

    但是不能就认为是100%等价.
    乐观悲观,是站在"预估锁冲突"的角度.
    重量轻量,则是站在"加锁开销"的角度

  3. 挂起等待锁 vs 自旋锁

    • 挂起等待锁: 属于是悲观锁/重量级锁的一种典型实现.当线程无法获取锁时,会选择主动让出CPU,并进入等待队列(通常是被操作系统挂起),直到 锁被释放并收到通知后才重新参与CPU调度 。
    • 自旋锁: 属于乐观锁/轻量级锁的一种典型实现.当线程无法获取锁时,会不停的检测锁是否被释放,一旦锁释放了,就立即有机会能够获取到锁

    轻量级锁,就是基于"自旋"的方式实现的(JVM内部,用户态代码实现的)
    重量级锁,就是基于"挂起等待锁"的方式实现的(调用操作系统api,在内核中实现的)

  4. 公平锁 vs 非公平锁

    • 公平锁: 其他线程按照先来后到的顺序来获取锁.
    • 非公平锁: 其他线程按照"概率均等"的方式来竞争锁.(概率不一定是数学上的严格均等)

    synchronized属于非公平锁.

  5. 可重入锁 vs 不可重入锁

    • 可重入锁: 一个线程,针对同一把锁,可以连续加锁两次以上.
    • 不可重入锁: 一个线程,针对同一把锁,不能连续加锁两次以上,否则会出现死锁问题.

    可重入锁的实现逻辑:

    1. 记录当前是哪个线程持有了这把锁.
    2. 在加锁的时候判定,当前申请锁的线程,是否是锁的持有者线程.
    3. 通过计数器,记录加锁的次数,从而确定何时真正释放锁.
  6. 读写锁
    读写锁把加锁操作分成两种情况:读加锁写加锁.

    • 如果,多个线程,同时读一个变量,此时没有线程安全问题.
    • 但是,一个线程读/一个线程写 或者 两个线程都写 就会产生问题.

    读写锁提供了两种加锁的api:加读锁加写锁.

    • 如果两个线程,都是按照读方式加锁,此时不会产生锁冲突.
    • 如果两个线程,都是加写锁,此时会产生锁冲突.
    • 如果一个线程是读锁,一个线程是写锁,也会产生锁冲突.

    虽然两种加锁的api不同,但是解锁的api是一样的.

    Java标准库提供了ReentrantReadWriteLock类,实现了读写锁.

    • ReentrantReadWriteLock.ReadLock类表示一个读锁,这个对象提供了lock/unlock方法进行加锁解锁.
    • ReentrantReadWriteLock.WriteLock类表示一个写锁,这个对象也提供了lock/unlock方法进行加锁解锁.

    其中:
    读加锁和读加锁之间,不互斥.
    写加锁和写加锁之间,互斥.
    读加锁和写加锁之间,互斥.

    synchronized不是读写锁.

synchronized

特点

synchronized有以下特点:

  1. 开始时是乐观锁, 如果锁冲突频繁, 就转换为悲观锁.
  2. 开始是轻量级锁实现, 如果锁被持有的时间较长,就转换成重量级锁.
  3. 自旋 or 挂起等待,自适应.
  4. 是非公平锁.
  5. 是可重入锁.
  6. 不是读写锁.

加锁过程

刚开始使用synchronized加锁,首先锁会处于"偏向锁"状态.
当遇到线程之间的锁竞争时,会升级到"轻量级锁".
进一步的统计出现的频次,次数达到一定程度后,会升级到"重量级锁".

在这里插入图片描述

上述锁升级的过程,主要是为了能够让synchronized适应不同的场景,降低程序员的使用负担~

上述锁升级的过程不可逆!

理解一下偏向锁:
偏向锁,不是真的加锁,而只是做一个标记.(标记的过程非常的轻量高效)

锁消除(编译器的优化策略)

编译器会对你写的synchronized代码,做出判定,判定这个地方是否需要加锁.
如果这里没有必要加锁,编译器就能够自动把synchronized给干掉.

虽然存在锁消除,但是咱们在写代码的时候,不能完全指望这个,最好不要无脑加锁.

锁粗化(编译器的优化策略)

要讲锁粗化,那就不得不提到锁的粒度.

在一个锁内,代码越多,粒度就越粗;代码越少,粒度就越细.
注意了,这里的"代码多",指的是执行过程中实际运行的代码行数.
在这里插入图片描述

锁粗化,就是把多个"细粒度"的锁,合并成"粗粒度"的锁.
在这里插入图片描述

锁粗化可以减少获取和释放锁的次数,从而降低锁带来的开销。
需要注意的是,锁粗化并不是适用于所有情况的优化策略。在某些情况下,锁粒度较细可能是必要的,以保证程序的正确性和性能。

本文到这里就结束啦~

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

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

相关文章

JavaScript程序结构

程序结构有三种:选择结构、循环结构 、顺序结构 一、选择结构 1、简介 根据条件进行判断,从而执行不同的操作,称为选择结构(分支结构),其实就是条件判断 选择结构的类型:if、switch 2、if结…

第十七篇——九变篇:紧扣战略重心,别跑题

目录 一、背景介绍二、思路&方案三、过程1.思维导图2.文章中经典的句子理解3.学习之后对于投资市场的理解4.通过这篇文章结合我知道的东西我能想到什么? 四、总结五、升华 一、背景介绍 九变种前面偏向宏观给讲解了九变的含义;这一篇通过更加微观的…

如何基于numpy和scipy实现曲面的最大梯度计算与显示

大家在做三维可视化研究过程中,经常需要做三维曲面的绘制和相交分析,在不知道三维曲面方程的情况下,如何基于曲面散点数据计算曲面的最大梯度点和梯度线的三维可视化是大家基于曲面分析研究中的重点关注的问题,本文在python环境下,基于numpy、pandas、scipy和matplotlib等…

MYSQL————联合查询

联合多个表进行查询 设计数据时把表进行拆分,为了消除表中字段的依赖关系,比如部分函数依赖,传递依赖,这时会导致一条SQL查出来的数据,对于业务来说是不完整的,我们就可以使用联合查询把关系中的数据全部查…

dk安装redis

docker search redis docker pull redis 由于我是使用的镜像 所以我的拉取命令就是 docker pull dockerpull.com/redis 拉下镜像之后,点击下面地址选择自己需要的 redis 版本的配置文件 redis/redis.conf at 6.2.6 redis/redis GitHubRedis is an in-memory data…

Python和Java及MATLAB和CUDA显微镜导图

🎯要点 交互式设备控制和图像处理图像背景和阴影校正可视化萤光团位置算法和读取光学图像读写转换显微镜图像生物医学细胞图像分析荧光图像算法计算亮度数据和模拟表征新型染料和缓冲液强度估计细菌图像分析扫描透射和高分辨率透射图像模拟多模态成像分割可视化透射…

python安装protobuf记录

需求背景 客户需要protobuf环境配置以及插件安装,我主要通过在pycharm里面配置参数,将问题解决! 导语 Python中的protobuf(Protocol Buffers)是一种用于序列化结构化数据的语言无关、平台无关、可扩展的机制。它能够让您在不同的应用程序和语…

搭建基于QT的TCP服务器与客户端

1、实现功能 1、服务器和客户端能够建立连接 2、服务器可以给客户端发送信息 3、客户端可以给服务器发送信息 2、server 2-1、widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> #include <QTcpSocket> QT_BEGIN_NA…

2024 年 IBM 量子开发者大会:等你来

在 2024 年 IBM Quantum™ 开发者大会上&#xff0c;与会者将获得 IBM Quantum 尖端工具和即将推出的路线图更新的独家、亲身预览&#xff0c;所有这些都围绕一个主题 — — Qiskit 的性能。 2024 年 IBM 量子开发者大会 在此申请 重要日期 7 月 24 日&#xff1a; 开放申请 8 …

leetcode 902. Numbers At Most N Given Digit Set

题目链接 Given an array of digits which is sorted in non-decreasing order. You can write numbers using each digits[i] as many times as we want. For example, if digits [1,3,5], we may write numbers such as 13, 551, and 1351315. Return the number of posit…

Docker续6:容器网络

1.bridge-utils 一个用于Linux系统的网络桥接工具集。它提供了一些命令行工具&#xff0c;帮助用户创建、管理和配置网络桥接。网络桥接是一种将多个网络接口连接在一起&#xff0c;以使它们能够作为单个网络段进行通信的技术。 bridge-utils 常用的命令包括&#xff1a; b…

大阪OSAKA分子泵电源TC163HTC203TC353TC523TC1104TC553TC1813手侧

大阪OSAKA分子泵电源TC163HTC203TC353TC523TC1104TC553TC1813手侧

六、泛型事件框架

一、创建一个BaseEventSO的 基类 写一个泛型事件《T》 启动事件?.Invoke 二、创建一个监听类 同样泛型匹配Event的监听事件 创建OnEable在其中注册事件&#xff1b;OnDisable中注销事件 写一个if语句进行判断是否为空 三、创建一个ObjectEvent 传递Object 原因&#xf…

axios发送post请求实例

在body中的数据格式又有两种&#xff0c;一种是 json 数据格式&#xff0c;另一种是 字符串。具体要用哪种格式取决于后端入参的格式。 如果后端接收json数据类型&#xff0c;post 的 headers 需要设置 { ‘content-type’: ’application/json’ }&#xff0c;传给后端的数…

初学流量分析

一、基本知识 比赛中提供一个包含流量数据的 PCAP 文件&#xff0c;有时候也会需要选手们先进行修复或重构传输文件后&#xff0c;再进行分析。 PCAP 这一块作为重点&#xff0c;复杂的地方在于数据包里充满着大量无关的流量信息&#xff0c;因此如何分类和过滤数据是参赛者需…

stm32f103c8t6引脚功能

STM32F103C8T6拥有丰富的引脚功能&#xff0c;主要包括广泛的GPIO引脚、多种通信接口和特定的调试及电源管理引脚。 STM32F103C8T6是一款基于ARM Cortex-M3内核的微控制器&#xff0c;广泛应用于各种嵌入式系统设计中。它具有72MHz的处理速度&#xff0c;搭载128KB的Flash内存…

行业寒冬下的职场生态:卷与痛的交响曲

在2024年的初春&#xff0c;当万物复苏的季节理应带来希望与机遇之时&#xff0c;职场却笼罩在一片前所未有的“寒冬”之中。经济的波动、技术的快速迭代以及市场需求的微妙变化&#xff0c;共同编织了一幅复杂而严峻的行业图景。工作岗位的数量锐减&#xff0c;质量下滑&#…

域名解析DNS服务

DNS服务基础知识 DNS&#xff1a;Domain Name System (域名系统) DNS 是互联网上解决网络中机器命名的一种系统。在网络中&#xff0c;一台主机去访问另外一台主机时&#xff0c;必须要 知道目标主机的IP地址&#xff0c;虽然网络上的节点都可以用IP地址来标识&#xff0c;并且…

Unity | 基础知识

装箱和拆箱&#xff1a; 获取对方的类型&#xff1a; GetType通过打点调用 as进行类型转换 运用Convert进行类型转换&#xff1a; 二维数组的定义 结构体类型 不同名称空间来调用&#xff1a;

构建开发全能型档期预约系统

档期预约系统作为一种高效的时间管理和资源分配工具&#xff0c;在现代社会中的应用越来越广泛。以下是关于档期预约系统的详细分析&#xff1a; 一、系统概述 档期预约系统旨在帮助各类服务机构&#xff08;如医疗机构、美容院、教育培训机构等&#xff09;优化服务流程&…