【并发编程】synchornized原理

       📝个人主页:五敷有你      
 🔥
系列专栏:并发编程
⛺️稳重求进,晒太阳


 

目录

Monitor概念

Java对象头

普通对象

数组对象

Monitor(锁)

Monitor结构如下:

注意:

原理之synchornized

        轻量级锁

锁膨胀(重量级锁)

自旋优化

偏向锁

示例

 轻量级锁与偏向锁加锁的对比

偏向状态

一个对象创建时:

撤销-调用对象hashCode

撤销-其他线程使用对象

批量重定向

批量撤销

锁消除


Monitor概念

Java对象头

以32位机的机器为例

普通对象

数组对象

其中Mark Word结构为

Monitor(锁)

Monitor被翻译为监视器或管程

每个Java对象都可以关联一个Monitor对象,如果使用synchronized给对象上锁(重量级)之后,该对象头的Mark Word中就被设置指向Monitor对象的指针。

Monitor结构如下:

  • 刚开始Monitor中Owner为null
  • 当Thread-2执行synchronized(obj) 就会将Monitor 的所有者Owner设置为Thread-2,Monitor中只能有一个Owner.
  • 在Thread-2上锁的过程中,如果Thread-3,Thread-4也来执行synchronized(obj) 就会进入EntryList BLOCKED阻塞队列
  • Thread-2执行完同步代码块的内容,然后唤醒EntryList 中等待的线程来竞争锁,竞争的锁是非公平的。
  • 图中WaitSet中的Thread-0,Thread-1 是之前获得过锁,但条件不满足进入WAITING 状态的线程。后面会将wait-notify

注意:

  • synchronized 必须是进入同一个对象的monitor才有上述的效果
  • 不加synchronized 的对象不会关联监视器,不遵从以上规则

原理之synchornized

        轻量级锁

        轻量级锁的使用场景:如果一个对象虽然有多线程访问,但多线程访问的时间是错开的(也就是没有竞争)。那么就可以使用轻量级锁优化

        轻量级锁对使用者是透明的,即语法仍然是synchronized

        假设有两个同步块,利用同一个对象加锁。

static final Object obj=new Object();
public static void method1(){
    synchornized(obj){
        //同步块 A
            method2();
    }
}
public static void method2(){
    synchronized(obj){
        //同步块b    
    }
}
  • 创建锁对象,每个线程的栈帧会包含一个锁记录的结构,内部可以存储锁定对象的Mark Word

  • 让锁记录中的Object reference指向锁对象,并尝试使用CAS替换Object的Mark Word,将Mark Word的值存入锁记录

  • 如果CAS替换成功,对象头中存储了锁记录地址和状态,表示由该线程给对象加锁,这时图示如下:

  • 如果cas失败:
    • 如果是其他线程已经持有了该Object的轻量级锁,这时候表面有竞争,进入锁膨胀过程。
    • 如果是自己执行了synchronized 锁重入,那么再添加一条Lock Record 作为重入的计数

  • 当退出synchronized 代码块(解锁时) 如果有取值为null 的记录,表示有重入,这时重置锁记录,表示重入计数-1

  • 当退出synchronized 代码块(解锁时)锁记录的值不为null ,这时使用cas将Mark Word的值恢复给对象头

成功:解锁成功

失败: 说明轻量级锁进行了锁膨胀或已经升级为重量级锁。进入重量级锁解锁流程

锁膨胀(重量级锁)

如果在尝试加轻量级锁的过程中,CAS操作无法成功,这时一种情况就是有其他线程为此对象加上了轻量级锁(有竞争),这时需要进行锁膨胀,将轻量级锁变为重量级锁。

当Thread -1进行轻量级加锁时,Thread-0已经对该对象加了轻量级锁

这时候Thread-1加锁失败,进入锁膨胀流程。

  • 即为Object对象申请Monitor锁,让Object指向重量级锁地址
  • 然后自己进入Monitor的EntryList BLOCKED

  • 当Thread-0 退出同步代码块时,使用CAS将MarkWord 的值恢复给对象头,失败,这时候会进入重量级解锁流程,按照Monitor地址找到Monitor对象,设置Owner为null, 唤醒EntryList 中的BLOCKED
自旋优化

自选在多核CPU下才有意义

重量级锁竞争的时候,还可以使用自旋来进行优化,如果当前线程自选成功(即这时候锁线程已经退出了同步块,释放了锁)),

这时当前线程就可以避免阻塞。

偏向锁

        轻量级锁在没有竞争时(就自己这个线程),每次重入都需要进行CAS操作。

        Java6 引入了偏向锁来进一步优化,只有第一次使用CAS将线程ID设置到对象的Mark Word头,之后发现这个线程ID是自己的就表示没有竞争,不需要重新CAS,以后只要不发生竞争,这个对象就归线程所有。

示例
static final Object obj=new Object();
public static void m1(){
    synchronized(obj){
        //同步代码块A
        m2();    
    }
}
public static void m2(){
    synchronized(obj){
        //同步代码块B
        m3();    
    }
}
public static void m3(){
    synchronized(obj){
        //同步块C    
    }
}
 轻量级锁与偏向锁加锁的对比

偏向状态

回忆一下对象头格式

一个对象创建时:

        如果开启了偏向锁,那么对象创建后,markword值为0x05即最后三位为101(表示可以偏向的状态),这时它的thread epoch age都为0

        偏向锁是默认是延迟的,不会在程序启动后立刻生效。如果想要避免延迟,可以加VM参数:xx:BiasedLockingstartupDalay=0来禁止延迟

        如果没有开启偏向锁,那么对象创建后,markword值为0x01,即后三位为001,这时它的hashcode 、age都为0,第一次用到hashcode时才会赋值。

(调用了hashcode会让偏向状态禁用,让他变为不可偏向的对象)

撤销-调用对象hashCode

调用了对象的hashCode,但偏向锁的对象MarkWord中存储的是线程id,如果调用hashCode会导致偏向锁被撤销

  • 轻量级锁会在锁记录中记录hashCode
  • 重量级锁会在Monitor中纪录hashCode

在调用hashCode后使用偏向锁,记得去掉-xx:UseBiassedLocking

输出

撤销-其他线程使用对象

        当有其他线程使用偏向锁时,会偏向锁升级为轻量级锁

批量重定向

        如果对象虽然被多个线程访问,但没有竞争,这时候偏向了线程T1的对象仍有机会重新偏向T2,重偏向会重置对象的ThreadID,

        当撤销偏向锁阈值超过20次后,jvm会认为,我是不是偏向错了,于是会在这些对象加锁时重新偏向加锁线程。

详解:

    我们知道,当我们使用synchronized关键字的时候,一个对象a只被一个对象访问的时候,对对象加的锁偏向锁,如果之后出现第二个线程访问a的时候(这里只考虑线程交替执行的情况,不存在竞争),不管线程1是已死亡还是运行状态,此时锁都会升级为轻量锁,并且锁升级过程不可逆。

    但是如果有很多对象,这些对象同属于一个类(假设是类A)被线程1访问并加偏向锁,之后线上2来访问这些对象(不考虑竞争情况),在通过CAS操作把这些锁升级为轻量锁,会是一个很耗时的操作。

JVM对此作了优化:

    当对象数量超过某个阈值时(默认20, jvm启动时加参数-XX:+PrintFlagsFinal可以打印这个阈值 ),Java会对超过的对象作批量重偏向线程2,此时前20个对象是轻量锁,后面的对象都是偏向锁,且偏向线程2。

批量撤销

当撤销偏向锁阈值超过40次后,jvm会这样觉得,我们确实偏向错了,根本就不该偏向,于是整个类的所有对象都会变为不可偏向的,新建的对象也是不可偏向的

锁消除

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C#MQTT编程08--MQTT服务器和客户端(cmd版)

1、前言 前面完成了winform版,wpf版,为什么要搞个cmd版,因为前面介绍了mqtt的报文结构,重点分析了【连接报文】,【订阅报文】,【发布报文】,这节就要就看看实际报文是怎么组装的,这…

基于Python的疫情返乡人员管理系统的设计与实现-计算机毕业设计源码11411

摘 要 近年来,随着经济全球化和社会发展,返乡现象日益普遍,其中部分返乡人员可能存在抗疫突发事件的风险,为此,本文从宏观层面探讨了疫情期间返乡人员的管理方案。首先,建立疫情返乡人员的信息登记系统;其次…

python爬虫代码示例:爬取京东详情页图片

python爬虫代码示例:爬取京东详情页图片 一、Requests安装及示例 爬虫爬取网页内容首先要获取网页的内容,通过requests库进行获取。 GitHub: https://github.com/requests/requests PyPl: https://pypi.python.org/pypi/requests 官方文档:http://wwwpython-requ…

KaiwuDB × 风电企业 | 高性能、低成本、释放数据价值

项目背景 某风电企业是国内一流的大型风电装备公司,其自主研发建设了新一代高标准风电 SCADA 系统,专门用于解决风场风机状态监控、发电监测、综合能源管控、智能化运管等难题。 现公司已承接多个风场的风机管理运营工作,共计包含 96 台风力…

最新企业数据实时同步软件推荐

实时同步软件能够帮助企业快速、准确地共享和更新数据,提高工作效率和决策质量。本文将介绍企业数据实时同步的概念、意义和应用场景,并推荐几款非常优秀的企业数据实时同步软件。 一、数据实时同步的意义 企业数据实时同步是指在企业内部或跨部门之间&…

分布式概念

文章目录 一、CAP定理和BASE定理1.1 CAP定理1.2 CAP取舍1.3 BASE定理 二、分布式事务2.1 柔性事务2.2 两阶段提交协议2.3 三阶段提交协议 三、分布式ID3.1 数据库自增ID3.2 数据库多主模式3.3 号段模式3.4 雪花算法3.5 Leaf3.6 使用Redis生成ID 四、限流算法4.1 固定窗口计数器…

使用的uview 微信高版本 头像昵称填写能力

<template><view><button class"cu-btn block bg-blue margin-tb-sm lg" tap"wxGetUserInfo">一键登录</button><view><!-- 提示窗示例 --><u-popup :show"show" background-color"#fff">&…

【踩坑日志】SpringBoot读取nacos配置信息并提取信息中的IP地址(配置属性解析异常+排错记录)

缘起 &#xff1a;项目需读取nacos中动态的TDengine数据库连接信息并提取IP&#xff0c;一个并不复杂的操作&#xff0c;但作为一个nacos知识浅薄的菜鸡&#xff0c;我愣是捯饬了几个小时……惭愧惭愧…… 异常代码 Data Component public class TaosLink { // Value("…

Docker--harbor私有仓库

目录 一、什么是Harbor&#xff1f; 二、Harbor的特性 三、Harbor的构成 四、部署 五、维护管理Harbor 一、什么是Harbor&#xff1f; Harbor 是 VMware 公司开源的企业级 Docker Registry 项目&#xff0c;其目标是帮助用户迅速搭建一个企业级的 Docker Registry 服务。 …

【昇思技术公开课笔记-大模型】Transformer理论知识

什么是Transformer Transformer是一种神经网络结构&#xff0c;由Vaswani等人在2017年的论文“Attention Is All You Need”中提出&#xff0c;用于处理机器翻译、语言建模和文本生成等自然语言处理任务。 Transformer与传统NLP特征提取类模型的区别主要在以下两点&#xff1…

WebGL开发智慧城市应用

在使用WebGL实现智慧城市应用时&#xff0c;需要考虑一系列的问题&#xff0c;以确保系统的性能、安全性和用户体验。以下是在开发WebGL智慧城市应用时需要注意的问题&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;…

【iOS】——基于Vision Kit框架实现图片文字识别

文章目录 前言一、文本识别的分类二、实现步骤1.导入Vision Kit框架2.创建请求处理器3.在请求处理器中设置文字识别功能4.将图片添加到请求处理器中5.发起文字识别请求6.处理识别结果 三、运行结果测试1.纯英文环境2.中英文混合环境 前言 根据苹果的官方文档&#xff0c;Visio…

MySQL面试题 | 15.精选MySQL面试题

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

YOLOv7全网独家首发:DCNv4更快收敛、更高速度、更高性能,效果秒杀DCNv3、DCNv2等 ,助力检测实现暴力涨点

💡💡💡本文独家改进:DCNv4更快收敛、更高速度、更高性能,完美和YOLOv7结合,助力涨点 DCNv4优势:(1) 去除空间聚合中的softmax归一化,以增强其动态性和表达能力;(2) 优化存储器访问以最小化冗余操作以加速。这些改进显著加快了收敛速度,并大幅提高了处理速度,DCN…

Mybatis 动态SQL条件查询①

需求 : 根据用户的输入情况进行条件查询 新建了一个 userInfo2Mapper 接口,然后写下如下代码,声明 selectByCondition 这个方法 package com.example.mybatisdemo.mapper; import com.example.mybatisdemo.model.UserInfo; import org.apache.ibatis.annotations.*; import j…

LeetCode、2462. 雇佣 K 位工人的总代价【中等,最小堆+双指针】

文章目录 前言LeetCode、2462. 雇佣 K 位工人的总代价【中等&#xff0c;最小堆双指针】题目及类型思路及代码实现 资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝2W&#xff0c;csdn博客专家、Java领域优质创作者&#xff0c;博客之星、阿里云平台优质作者、专注于Java后…

【安全篇 / FortiGuard】(7.4) ❀ 02. 独立VDOM下的FortiGuard服务升级 ❀ FortiGate 防火墙

【简介】由于业务的需要&#xff0c;创建两个独立VDOM&#xff0c;每个VDOM有各自的宽带&#xff0c;但是FortiGuard服务却无法升级&#xff0c;有什么办法解决吗&#xff1f; VDOM概念 首先我们看看什么是VDOM。 ① VDOM将你的FortiGate划分为多个逻辑设备&#xff0c;并将一个…

用 Python 制作可视化 GUI 界面,一键实现自动分类管理文件!

经常杂乱无章的文件夹会让我们找不到所想要的文件&#xff0c;因此小编特意制作了一个可视化GUI界面&#xff0c;通过输入路径一键点击实现文件分门别类的归档。 不同的文件后缀归类为不同的类别 我们先罗列一下大致有几类文件&#xff0c;根据文件的后缀来设定&#xff0c;大…

Babylonjs inspector工具开启embedMode模式后不显示

项目地址见&#xff1a;https://github.com/tipace/simple-babylonjs 简单的babylonjs example 本身问题挺简单的&#xff0c;仅做一个记录。开始以为是babylon的问题&#xff0c;最后发现是css问题。 因为是做demo&#xff0c;把canas设置为占满全屏&#xff0c;习惯性的写…

Linux--磁盘与文件系统

目录 1.什么是文件系统 2.磁盘 2.1什么时磁盘 2.2磁盘的物理存储结构 2.3磁盘的逻辑抽象结构 3.磁盘文件系统&#xff08;EXT2&#xff09; inode Table(i结点表) Data Block inode Bitmap(inode位图) Block Bitmap(块位图) 在Linux如何删除文件 Group Descriptor Ta…