synchronized锁

synchronized
  • 类锁:给类的静态方法加上synchronized 关键字进行修饰,
    • 锁的是当前类class,一个静态同步方法拿到锁,其他静态同步方法就会等待
    • 静态同步方法和普通同步方法间是没有竞争的
  • 对象锁:给类的方法加上synchronized 关键字进行修饰
    • 锁的是当前对象 this,如果一个对象里有多个synchronized 方法,某个时刻只能有一个线程去调用这个对象其中的一个方法
    • 没有加synchronized的方法不受影响
    • 如果是不同的对象,锁就不同了,也就不会互相干扰
  • 同步代码块锁
    • 锁的是 synchronized () 内的对象
  • 加锁可以保证线程安全,但是会带来性能的下降,在高并发下,能不加锁就不要加锁,一定要加锁也要使加锁的粒度尽可能的小

为什么每一个对象都可以成为锁

  • 在hotsport虚拟机中,monitor采用的是ObjectMonitor实现的,每一个对象天生都带着一个监视器对象,每一个被锁住的对象都会和monitor关联起来
  • monitor 的本质是依赖于操作系统的 Mutex Lock 实现,操作系统实现线程的切换需要在用户态和内核态之间切换,成本很高
  • ObjectMonitor对象重要的属性
    • owner属性记录了持有ObjectMonitor对象的线程id
    • count 初始值为0,表示当前锁对象是否被锁定,加锁就加1,释放锁就减1
    • recursions 初始值为0,表示重入次数
    • entryList 阻塞队列,用于存放阻塞的线程
    • waitSet 等待队列,存放等待的线程

synchronized

  • 由对象头中的 mark word 根据锁标志位的不同来表示锁的状态
  • 在java5之前,只有synchronized ,是操作系统级别的重量级操作,涉及到用户态和内核态的切换,如果锁竞争激烈,性能下降严重
    • java的线程是映射到操作系统的原生线程之上的,如果要阻塞或者唤起一个线程就需要操作系统的接入,就需要在用户态和内核态之间切换
    • 这种切换是很消耗系统资源的,因为用户态和内核态都有自己专用的内存空间、寄存器等,用户态切换至内核态需要传递很多参数和变量给内核,内核也需要保存好用户态的一些变量,以便内核态调用结束后切换回用户态继续工作
    • 所以如果同步代码块中的内容很简单,有可能用户态和内核态之间的切换时间比代码本身的执行时间还长
  • 所以 java6之后,通过引入轻量级锁和偏向锁,来减少获得锁和释放锁所带来的性能消耗,从而优化了synchronized
synchronized 的锁升级
  • 无锁,对象新建出来,还没有和任何synchronized关联,就是无锁的状态
  • 偏向锁:mark word 前54位存储偏向的线程id
  • 轻量级锁
  • 重量级锁
  • 锁升级的过程就是,先cas自旋,实在得不到再阻塞

流程如下:
在这里插入图片描述

偏向锁
  • 一个 synchronized 方法被一个线程抢到了锁,这个方法所在的对象就会在 mark word 中修改偏向锁的标志位,同时前54位也会用来存储线程指针,也就是偏向线程id

  • 偏向模式,如果不存在其他线程竞争,那么持有偏向锁的线程永远不需要进行同步,也就是说,一段同步代码块,如果一直被一个线程多次访问,那么该线程后续的访问会自动获得锁

  • 因为HotSpot作者研究发现,多线程的情况下,大多数时候,锁不仅不存在竞争关系,还存在锁由同一个线程多次获得的情况

  • 所以只需要锁在第一次被拥有的时候,记录下线程的id,这样偏向线程会一直持有锁,这个线程后续进入和退出听不代码块的时候,不需要再次加锁和释放锁,而是去检查 mark word 中的偏向线程id是不是自己

    • 如果是那么锁偏向于当前线程,就不需要再去尝试获得锁了,会直接进入同步块,不需要每次都通过CAS更新对象头,如果自始至终都只有一个线程持有锁,那么偏向锁几乎没由额外的开销,性能极高
    • 如果偏向线程id不是当前线程,表示发生了竞争,表示锁已经不是总偏向于一个线程了,这个时候,会尝试使用CAS来更新 mark word 里的线程id为当前线程的id
      • 如果CAS竞争成功, mark word 里的线程id 就会替换为当前线程的id,锁也不会升级,仍然是偏向锁,只不过是从一个线程偏向到另一个线程
      • 但是如果CAS竞争失败,这个时候就有可能需要 撤销偏向锁,升级为轻量级锁,使线程间公平竞争
  • 偏向锁会偏向于第一个访问到锁的线程,且只有偏向锁被其他线程竞争,持有偏向锁的线程才会释放锁,否则线程是不会主动释放锁的,而对于持有偏向锁的线程也就不需要触发同步,就能在没有资源竞争的情况下消除了同步语句

  • jdk6之后,默认就开启了偏向锁

    • 但是 HotSpot 虚拟机在启动后有个 4s 的延迟才会对每个新建的对象开启偏向锁模式,这四秒钟之内默认会进入轻量级锁
    • 因为 JVM 启动时会进行一系列的复杂活动,比如装载配置,系统类初始化等等。在这个过程中会使用大量synchronized关键字对对象加锁,且这些锁大多数都不是偏向锁。为了减少初始化时间,JVM默认延时加载偏向锁

偏向锁的撤销

  • 只有发生竞争时,偏向锁才会释放,原本持有偏向锁的线程才会被撤销
  • 撤销需要等待全局安全点,也就是该时间点上没有字节码正在执行,同时检查持有偏向锁的线程是否还在执行
  • 如果线程正在执行同步方法,则升级锁
    • 其他线程就尝试使用CAS来更新 mark word 里的线程id,从而抢夺锁,偏向锁就会被取消掉并升级为轻量级锁
    • 轻量级锁仍然由原本的线程A持有,A会继续执行同步代码,其他正在竞争的线程会进入自旋等待重新获取轻量级锁
  • 如果线程执行完了,就会释放锁,并将对象头设置为无锁状态,并撤销偏向锁,被其他线程抢占,重新偏向

java15后会逐步废弃偏向锁

  • 在java15之前,偏向锁是默认开启的
  • 但是15之后,就默认不在开启了,除非手动开启
轻量级锁
  • 关闭偏向锁功能或者多线程竞争偏向锁都会导致偏向锁升级为轻量级锁,升级为轻量锁,会把偏向锁标记改为0,并设置标志位为00

  • 对象头 mark word 前62位用来记录线程的id,后两位为锁的标志位 00

  • 轻量级锁在没有多线程的竞争下,通过cas来代替重量级锁,较少性能的消耗,能够在线程近乎交替执行同步代码块时提高性能

  • 轻量级锁升级过程

    • 首先线程A拿到锁,这时候的锁是偏向锁,偏向于A
    • 线程B又来抢夺锁,发现锁对象头的 当前线程id 不是自己,线程B就会通过CAS操作去修改线程id希望能获得锁
    • 如果B获得成功,也就是A已经执行完了 ,B会把 mark word 当前线程id设置为自己,锁仍然是偏向锁,只是重新偏向于B
    • 如果B获取失败,也就是锁仍然被其他线程占用,锁就会升级为轻量锁,轻量级锁仍然由之前持有锁的线程继续持有,B线程会自旋等待获取轻量级锁

轻量级锁的加锁

  • JVM会为每个线程在当前线程的栈帧中创建用于存储锁记录的空间,称为 Displaced Mark word
  • 如果一个线程获得锁时发现是轻量级锁,会把当前锁的 Mark word 复制到自己的 Displaced Mark word 里
  • 然后线程尝试用CAS将锁的 Mark word 替换为指向锁记录的指针,
    • 如果成功,就代表获取到了锁,
    • 如果失败,表示 Mark word 已经被替换为了其他线程的锁记录,说明存在其他线程竞争锁,当前线程就会尝试使用自旋来获取锁

轻量级锁的释放

  • 释放锁时,当前线程会使用CAS将 Displaced Mark word 的内容复制到锁的 Mark word 中
  • 如果没有发生竞争这个复制操作就会成功,如果有其他线程因为自旋多次导致轻量级锁升级为了重量级锁,那么CAS操作会失败,此时会释放锁并释放被阻塞的线程

轻量级锁和偏向锁的区别:

  • 偏向锁是没有竞争关系的,轻量级锁存在锁的竞争,竞争失败,会自旋尝试抢占锁
  • 偏向锁只有竞争发生才会释放锁,轻量级锁每次退出同步代码块时都需要释放锁
重量级锁
  • 对象头 mark word 前62位用来指向互斥量 (重量级锁) 的指针,后两位为锁的标志位 10

  • 当线程自旋达到一定次数,仍然没有获得锁,也就是有大量线程在竞争锁,那么就会升级锁为重量级锁

  • jdk6之前默认是自旋次数达到10次,或者自旋线程数超过cpu核数的一半,都会升级为重量级锁

  • jdk7增加了自适应自旋锁,也就是自旋的次数变的不在固定,

    • 通过同一个锁上一次自旋的时间,和拥有锁线程的状态来决定
    • 也就是如果线程自旋成功了,那么下次自旋的最大次数就会增加,因为JVM认为上次成功了,那么这次也有很大概率成功
    • 反之如果很少会自旋成功,那么下次就会减少自旋的次数甚至不自旋,来避免cpu空转

synchronized

  • 对于同步代码块

    • 一般情况下,一把锁,会有一个monitorenter指令,和两个monitorexit指令

    • 加锁会执行monitorenter

    • 释放锁会执行monitorexit,如果产生异常也会执行monitorexit,所以synchronized产生异常也可以释放锁

  • 对于同步方法

    • 会加上 ACC_SYNCHRONIZED 标识,代表这个方法是同步方法
    • 如果方法持有ACC_SYNCHRONIZED 标识,执行前就会去获取 monitor ,执行完再释放monitor
  • 对于静态同步方法

    • 会加上 ACC_SYNCHRONIZED 和 ACC_STATIC 标识,用于区分类锁和对象锁

synchronized 加锁流程

  • 当执行monitorenter时,
    • 如果锁计数器为0,就说明锁没有被其他线程持有,虚拟机会将当前线程设置为锁的持有线程,并且把锁计数器加1,重入次数加1,然后执行同步代码块的业务代码
    • 如果锁计数器不为0
      • 且持有线程是当前线程,虚拟机会把锁计数器加1,
      • 如果不是,就需要进入当前锁对象的阻塞队列,等待其他线程释放锁
  • 当执行monitoreixt时,虚拟机会把锁计数器减1,当锁计数器为0时,会擦除锁的持有线程,这样就释放了锁

锁升级到轻量级锁,重量级锁后,mark word中保存的就分别是线程栈帧里的锁记录指针和重量级锁指针,不在保存hashcode 和GC的年龄,这些信息的去向是:

  • 首先,java中一个对象如果计算过一次哈希码,就应该保持这个值不变,除非用户手动重载hashcode方法就可以返回任意值,如果哈希码经常变动,会导致很多依赖哈希码的对象都存在风险

  • 绝大多数对象的哈希码都来自于 object 的 hashcode 方法,通过在对象头中存储计算结果来保证第一次计算过后,在次调用hashcode 方法取到的哈希码就永远不会改变

  • 无锁状态下,mark word 中可以存储对象的 hash code 值,当对象的 hashcode 方法第一次被调用调用时,JVM就会生成对应的hash code值并存储到 mark word 中

  • 对于偏向锁,在线程获取偏向锁时,会使用线程id和epoch 值覆盖 hash code值所在的位置,所以如果一个对象已经计算过了哈希码,这个对象就无法被设置为偏向锁

    • 因为如果允许的话,会导致hash code 值和线程id相互覆盖,导致前后调用hashcode 方法的计算结构不一致,所以偏向锁和哈希码不共存
    • 所以如果一个对象处于偏向锁状态(锁是偏向锁,但是已经释放过了锁),被调用hashcode方法后,会直接膨胀为轻量级锁
    • 如果一个对象处于偏向锁过程中(锁是偏向锁,且没有释放锁),被调用hashcode方法后,会直接膨胀为重量级锁
  • 而对于轻量级锁,JVM会在当前线程的栈帧中创建一个锁记录的空间,用于存放锁对象的mark word 拷贝,这个拷贝中就包含了hash code值和GC的年龄,释放锁后会将这些信息写回对象头

  • 升级到重量级锁后,mark word 保存了重量级锁的ObjectMonitor 类里有字段记录非加锁状态下的mark word ,锁释放后信息也会被写回到对象头

synchronized 锁的优缺点

  • 偏向锁的优点是:加锁和解锁不需要额外的消耗,和执行非同步方法相比仅存在纳秒级的差距,缺点是如果线程间存在竞争,会带来额外的锁撤销的消耗,所以只适用于只有一个线程访问同步块的场景
  • 轻量级锁的优点是:即使存在线程的竞争也不会阻塞,提高了程序的响应速度,缺点是始终拿不到锁的线程自旋会消耗cpu,适用于追求响应时间,同步块执行时间很短的场景
  • 重量级锁,线程间的竞争不会消耗cpu去自旋,缺点是线程阻塞会导致响应缓慢,适用于追求吞吐量,同步块执行时间较长的情况

锁消除(同步省略)

  • 在动态编译同步块的时候,JIT编译器可以借助逃逸分析来判断,同步块锁,是否有加锁的必要,如果没有就可以不考虑同步,也就是所谓的锁消除
  • 因为加锁的代价是很高的,消除锁可以大大提高并发性和性能,这种情况字节码文件依然会有加锁操作,但是执行的时候会去掉
    public static void test4() {
        Object obj =new Object();
        //例如这里,每个线程进来都会new一个对象,每个线程都有一个锁,没有意义
        synchronized (obj){
            System.out.println("锁消除案例");
        }
    }

锁粗化

  • 如果对同一个对象执行了连续的加锁和解锁的操作,那么 JIT 会将多个连续的加锁、解锁操作连接在一起,扩展成一个范围更大的锁,避免频繁的加锁和解锁
    static Object lock =new Object();
    
    public static void test5() {

        new Thread(()->{
            synchronized (lock){
                System.out.println("业务1");
            }
            synchronized (lock){
                System.out.println("业务2");
            }
            synchronized (lock){
                System.out.println("业务3");
            }
        },"锁粗化案例").start();
    }

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

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

相关文章

权威外媒聚焦:Messari强调波场TRON在全球加密支付领域的引领作用

近日,金融时报、费加罗报及美联社等海外权威媒体就波场TRON 在全球加密支付领域的重要进展发布了相关报道。报道引述加密研究机构Messari 《Crypto Theses for 2024》年度报告,重点强调了波场TRON在推动全球加密货币支付尤其是稳定币USDT应用方面的显著成就。 报道提到,波场TR…

你的第一个C/S程序

目录 socket服务端代码客户端代码执行结果 socket socket基础知识 服务端代码 import socket import threading import timeMSG_LENGTH 64 DISCONNECTED !CONNECTION CLOSED connections 0#定义服务器地址 server_ip socket.gethostbyname(socket.gethostname()) server…

三城三奖!苏州金龙助力各地公共交通打造高品质线路

元旦前夕,由中国交通报社主办的绿色运输可持续发展座谈会暨2023年度“新能源公交高品质线路”经验交流会在京举行,来自全国各地的100余名行业管理部门、公交客运企业代表参会。会上同时评选出20条各具特色的“新能源公交高品质线路”及6家“我的公交我的…

后端主流框架-SpringMvc-day2

Java中的文件下载 2 文件下载 文件下载:就是将服务器(表现在浏览器中)中的资源下载(复制)到本地磁盘; 2.1 前台代码 前台使用超链接,超链接转到后台控制器,在控制器通过流的方式…

阿里云服务器(ECS云服务器)安装redis

前言: 笔者使用的是云服务器是阿里云的ECS服务器 这个服务器内核是Alibaba Cloud Linux 3。 使用的命令行工具为Alibaba Could Manager 命令行工具连接服务器这里就不多说了,如果没有用过的小伙伴可以去看阿里云的官方文档,很详细。 下面…

【51单片机系列】LCD1602液晶模块

本文是关于液晶显示屏的相关介绍。相对于静态数码管、动态数码管、LED点阵等,LCD1602液晶显示器能够显示更多的字符数字信息,并且也是常用的一种显示装置。 文章目录 一、LCD1602介绍1.1、LCD1602简介1.2、LCD1602常用指令1.3、LCD1602使用 二、LCD1602使…

[雷池WAF]长亭雷池WAF配置基于健康监测的负载均衡,实现故障自动切换上游服务器

为了进一步加强内网安全,在原有硬WAF的基础上,又在内网使用的社区版的雷池WAF,作为应用上层的软WAF。从而实现多WAF防护的架构。 经过进一步了解,发现雷池WAF的上游转发代理是基于Tengine的,所以萌生出了一个想法&…

SpringMVC-获取请求参数

1. 通过ServletAPI获取请求参数 /**** param request HttpServletRequest对象,直接作为形参传入方法,前端处理器就是一个Servlet* 所以前端处理器可以获得HttpServletRequest对象,并根据控制器方法的形参将对象传递给方法* re…

勒索事件急剧增长,亚信安全发布《勒索家族和勒索事件监控报告》

近期(12.15-12.21)态势快速感知 近期全球共发生了247起攻击和勒索事件,勒索事件数量急剧增长。 近期需要重点关注的除了仍然流行的勒索家族lockbit3以外,还有本周top1勒索组织toufan。toufan是一个新兴勒索组织,本周共发起了108起勒索攻击&a…

一文读懂$mash 通证的 “Fair Launch” 规则,将公平发挥极致

Solmash 是Solana生态中由社区主导的铭文资产LaunchPad平台,该平台旨在为Solana原生铭文项目,以及通过其合作伙伴SoBit跨链桥桥接到Solana的Bitcoin生态铭文项目提供更广泛的启动机会。有了Solmash,将会有更多的Solana生态的铭文项目、资产通…

【JUC的四大同步辅助类】

文章目录 一、CountDownLatch二、CyclicBarrier三、Semaphore四、Phaser 提示:以下是本篇文章正文内容,下面案例可供参考 一、CountDownLatch CountDownLatch如同火箭发射,计数只能不断减减,当到达0时即发射 场景示例&#xff1…

elect函数可以设置等待时间,

欢迎关注博主 Mindtechnist 或加入【智能科技社区】一起学习和分享Linux、C、C、Python、Matlab,机器人运动控制、多机器人协作,智能优化算法,贝叶斯滤波与Kalman估计、多传感器信息融合,机器学习,人工智能&#xff0c…

ssm基于BS的仓库在线管理系统的设计与实现论文

摘 要 如今的时代,是有史以来最好的时代,随着计算机的发展到现在的移动终端的发展,国内目前信息技术已经在世界上遥遥领先,让人们感觉到处于信息大爆炸的社会。信息时代的信息处理肯定不能用之前的手工处理这样的解决方法&#x…

OpenCV-Python(23):傅里叶变换

原理 傅里叶变换是一种数学变换,用于将一个函数(在图像处理中通常是图像)从时域(空域)转换到频域。它将函数表示为一系列正弦和余弦函数的和,用于分析信号的频率和相位信息。 傅里叶变换的原理是将一个连续…

【iOS安全】JS 调用Objective-C中WKWebview Handler的三种方式

有三种实现途径 1. WKScriptMessageHandler OC部分:注册并实现Handler 将OC中的方法"nativeMethod"注册为JavaScript Message Handler,从而WebView中的JavaScript代码可以调用该方法 // Register in Objective-C code - (void)setupWKWebVi…

No Magic—复杂机电产品系统架构开发套件

产品概述 CATIA Magic,原名MagicDraw,俗称No Magic,被达索收购后融入3DExperience产品协同研发管理平台中,形成更具协同体验的系统工程解决方案。该软件提供对SysML/UML/UAF语言的完整支持,提供独有的MagicGrid方法论&…

5分钟了解接口测试

接口测试是指对系统接口进行测试的一种质量保障手段,主要是验证接口的功能、性能、安全性等方面是否符合预期。 在接口测试中,可以测试以下内容: 功能测试:验证接口的输入和输出是否符合预期,包括参数的正确性、返回结…

【无标题】idea的lombok插件支持@SuperBuilder注解啦

在我的博客阅读本文 1. 前言 今早进公司打开idea,弹出更新提示,简单看了下,原来是idea的lombok插件更新了,惊喜的发现update log上写着Add support for SuperBuilder。 为什么说是惊喜呢?因为之前也有用到这个的场景…

Go后端开发 -- Go Modules

Go后端开发 – Go Modules 文章目录 Go后端开发 -- Go Modules一、什么是Go Modules?二、GOPATH的工作模式1.GOPATH模式2.GOPATH模式的弊端 三、Go Modules模式创建项目1.go mod命令2.go mod环境变量3.使用Go Modules初始化项目4.修改模块的版本依赖关系 四、Go Modules下impo…

数据库:基础SQL知识+SQL实验2

(1)基础知识: 1.JOIN(连接): 连接操作用于根据指定的条件将两个或多个表中的数据行合并在一起。JOIN 可以根据不同的条件和方式执行,包括等值连接、不等值连接等。 (1&#xff09…