11、Synchronized与锁升级

1、一些面试题

谈谈你对Synchronized的理解

Synchronized的锁升级你聊聊

Synchronized的性能是不是一定弱于Lock

 

 

2、本章路线总纲

synchronized 锁优化的背景 

用锁能够实现数据的 安全性 ,但是会带来 性能下降 。 

无锁能够基于线程并行提升程序性能,但是会带来 安全性下降 。   

求平衡???   

synchronized锁:由对象头中的Mark Word根据锁标志位的不同而被复用及锁升级策略

3、Synchronized的性能变化

3.1、java5以前,只有Synchronized,这个是操作系统级别的重量级操作

重量级锁,假如锁的竞争比较激烈的话,性能下降

Java5之前,用户态和内核态之间的切换

 

 java的线程是映射到操作系统原生线程之上的,如果要阻塞或唤醒一个线程就需要操作系统介入,需要在户态与核心态之间切换,这种切换会消耗大量的系统资源,因为用户态与内核态都有各自专用的内存空间,专用的寄存器等,用户态切换至内核态需要传递给许多变量、参数给内核,内核也需要保护好用户态在切换时的一些寄存器值、变量等,以便内核态调用结束后切换回用户态继续工作。 

在Java早期版本中, synchronized属于重量级锁,效率低下,因为监视器锁(monitor)是依赖于底层的操作系统的Mutex Lock来实现的 ,挂起线程和恢复线程都需要转入内核态去完成,阻塞或唤醒一个Java线程需要操作系统切换CPU状态来完成,这种状态切换需要耗费处理器时间,如果同步代码块中内容过于简单,这种切换的时间可能比用户代码执行的时间还长”,时间成本相对较高,这也是为什么早期的synchronized效率低的原因 

Java 6之后,为了减少获得锁和释放锁所带来的性能消耗, 引入了轻量级锁和偏向锁 

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

3.2.1、markOop.hpp

 

Monitor 可以理解为一种同步工具,也可理解为一种同步机制,常常被描述为一个 Java 对象。 Java 对象是天生的 Monitor ,每一个 Java 对象都有成为 Monitor 的潜质,因为在 Java 的设计中 ,每一个 Java 对象自打娘胎里出来就带了一把看不见的锁,它叫做内部锁或者 Monitor 锁。   

 

Monitor 的本质是依赖于底层操作系统的 Mutex Lock 实现,操作系统实现线程之间的切换需要从用户态到内核态的转换,成本非常高。   

3.2.2、Monitor(监视器锁)

 

Mutex Lock    

Monitor 是在 jvm 底层实现的,底层代码是 c++ 。 本质是依赖于底层操作系统的 Mutex Lock 实现,操作系统实现线程之间的切换需要从用户态到内核态的转换,状态转换需要耗费很多的处理器时间成本非常高。 所以 synchronized 是 Java 语言中的一个重量级操作。    

   

Monitor  java 对象以及线程是如何关联      

1. 如果一个 java 对象被某个线程锁住,则该 java 对象的 Mark Word 字段中 LockWord 指向 monitor 的起始地址   

2.Monitor 的 Owner 字段会存放拥有相关联对象锁的线程 id  

  

Mutex Lock 的切换需要从用户态转换到核心态中,因此状态转换需要耗费很多的处理器时间。   

3.3、java6开始,优化Synchronized

Java 6之后,为了减少获得锁和释放锁所带来的性能消耗,引入了轻量级锁和偏向锁

需要有个逐步升级的过程,别一开始就捅到重量级锁

4、synchronized锁种类及升级步骤

4.1、多线程访问情况,3种

只有一个线程来访问,有且唯一Only One

有2个线程A、B来交替访问

竞争激烈,多个线程来访问

4.2、升级流程

synchronized用的锁是存在Java对象头里的Mark Word中

锁升级功能主要依赖MarkWord中锁标志位和释放偏向锁标志位

64位标记图再看

 

4.3、无锁

package com.atguigu.juc.test;



import org.openjdk.jol.info.ClassLayout;



/**

 * @author shizan

 * @Classname MyObject

 * @Description TODO

 * @Date 2022/6/7 8:40 下午

 */

public class MyObject {

    public static void main(String[] args) {

        Object o = new Object();

        System.out.println("10 进制 hash 码: " + o.hashCode());

        System.out.println("16 进制 hash 码: " + Integer.toHexString(o.hashCode()));

        System.out.println("2 进制 hash 码: " + Integer.toBinaryString(o.hashCode()));

        System.out.println(ClassLayout.parseInstance(o).toPrintable());

    }

}

程序不会有锁的竞争

 

4.4、偏锁

4.4.1、主要作用

当一段同步代码一直被同一个线程多次访问,

由于只有一个线程那么该线程在后续访问时便会自动获得锁

同一个老顾客来访,直接老规矩行方便

看看多线程卖票,同一个线程获得体会一下

Hotspot 的作者经过研究发现,大多数情况下:   

多线程的情况下,锁不仅不存在多线程竞争,还存在锁 由同一线程多次获得的情况 , 

偏向锁就是在这种情况下出现的,它的出现是为了解决 只有在一个线程执行同步时提高性能 。 

 

通过CAS方式修改markword中的线程ID

4.4.2、偏向锁的持有

理论落地:   

       在实际应用运行过程中发现, “ 锁总是同一个线程持有,很少发生竞争 ” ,也就是说 锁总是被第一个占用他的线程拥有 , 这个线程就是锁的偏向线程 。   

       那么只需要在锁第一次被拥有的时候,记录下偏向线程 ID 。这样偏向线程就一直持有着锁 ( 后续这个线程进入和退出这段加了同步锁的代码块时, 不需要再次加锁和释放锁 。而是直接比较对象头里面是否存储了指向当前线程的偏向锁 ) 。   

如果相等 表示偏向锁是偏向于当前线程的,就不需要再尝试获得锁了,直到竞争发生才释放锁。以后每次同步,检查锁的偏向线程 ID 与当前线程 ID 是否一致,如果一致直接进入同步。无需每次加锁解锁都去 CAS 更新对象头。 如果自始至终使用锁的线程只有一个 ,很明显偏向锁几乎没有额外开销,性能极高。   

       假如不一致意味着发生了竞争,锁已经不是总是偏向于同一个线程了,这时候可能需要升级变为轻量级锁,才能保证线程间公平竞争锁。 偏向锁只有遇到其他线程尝试竞争偏向锁时,持有偏向锁的线程才会释放锁,线程是不会主动释放偏向锁的。   

  

技术实现:   

一个 synchronized 方法被一个线程抢到了锁时,那这个方法所在的对象就会在其所在的 Mark Word 中将偏向锁修改状态位,同时还会有占用前 54 位来存储线程指针作为标识。若该线程再次访问同一个 synchronized 方法时,该线程只需去对象头的 Mark Word 中去判断一下是否有偏向锁指向本身的 ID ,无需再进入 Monitor 去竞争对象了。   

细化案例Account对象举例说明

偏向锁的操作不用直接捅到操作系统,不涉及 用户到内核转换 , 不必要直接升级为最高级 ,我们以一个account对象的“对象头”为例, 

 

假如有一个线程执行到synchronized代码块的时候,JVM使用CAS操作把线程指针ID记录到Mark Word当中,并修改标偏向标示,标示当前线程就获得该锁。锁对象变成偏向锁(通过CAS修改对象头里的锁标志位),字面意思是“偏向于第一个获得它的线程”的锁。执行完同步代码块后,线程并不会主动释放偏向锁。 

这时线程获得了锁,可以执行同步代码块。当该线程第二次到达同步代码块时会判断此时持有锁的线程是否还是自己(持有锁的线程ID也在对象头里),JVM通过account对象的Mark Word判断:当前线程ID还在,说明还持有着这个对象的锁,就可以继续进入临界区工作。 由于之前没有释放锁,这里也就不需要重新加锁 。  如果自始至终使用锁的线程只有一个 ,很明显偏向锁几乎没有额外开销,性能极高。 

  

结论:JVM不用和操作系统协商设置Mutex(争取内核),它只需要记录下线程ID就标示自己获得了当前锁,不用操作系统接入。 

上述就是偏向锁:在没有其他线程竞争的时候,一直偏向偏心当前线程,当前线程可以一直执行。 

4.4.3、偏向锁JVM命令

java -XX:+PrintFlagsInitial |grep BiasedLock*

重要参数说明

 

*  实际上偏向锁在 JDK1.6 之后是默认开启的,但是启动时间有延迟,

*  所以需要添加参数 -XX:BiasedLockingStartupDelay=0 ,让其在程序启动时立刻启动。

*

*  开启偏向锁: 

* -XX:+UseBiasedLocking -XX:BiasedLockingStartupDelay=0

*

*  关闭偏向锁:关闭之后程序默认会直接进入---------------->>>>>>>>   轻量级锁状态。 

* -XX:-UseBiasedLocking

4.4.4、Code演示

一切默认

演示无效果

因为参数系统默认开启

-XX:+UseBiasedLocking                        开启偏向锁(默认)            

-XX:-UseBiasedLocking                        关闭偏向锁 

-XX:BiasedLockingStartupDelay=0              关闭延迟(演示偏向锁时需要开启) 

参数说明:

偏向锁在JDK1.6 以上默认开启 ,开启后程序启动几秒后才会被激活,可以使用JVM参数来关闭延迟 -XX:BiasedLockingStartupDelay=0 

如果确定锁通常处于竞争状态 则可通过JVM参数  -XX:-UseBiasedLocking  关闭偏向锁,那么默认会进入轻量级锁 

关闭延时参数,启用该功能 -XX:BiasedLockingStartupDelay=0

好日子终会到头......o(╥﹏╥)o   开始有第2个线程来抢夺了

4.4.5、偏向锁的撤销

当有另外线程逐步来竞争锁的时候,就不能再使用偏向锁了,要升级为轻量级锁

竞争线程尝试CAS更新对象头失败,会等待到全局安全点(此时不会执行任何代码)撤销偏向锁。

撤销

偏向锁的撤销   

偏向锁使用一种等到 竞争出现才释放锁的机制 ,只有当其他线程竞争锁时,持有偏向锁的原来线程才会被撤销。   

撤销需要等待全局安全点 ( 该时间点上没有字节码正在执行 ) ,同时检查持有偏向锁的线程是否还在执行:   

  

①   第一个线程正在执行 synchronized 方法 ( 处于同步块 ) ,它还没有执行完,其它线程来抢夺,该偏向锁会被取消掉并出现 锁升级 。   

此时轻量级锁由原持有偏向锁的线程持有,继续执行其同步代码,而正在竞争的线程会进入自旋等待获得该轻量级锁。   

②   第一个线程执行完成 synchronized 方法 ( 退出同步块 ) ,则将对象头设置成无锁状态并撤销偏向锁,重新偏向   。   

4.5、轻锁

4.5.1、主要作用

有线程来参与锁的竞争,但是获取锁的冲突时间极短

本质就是自旋锁

4.5.2、轻量级锁的获取

轻量级锁是为了在线程 近乎交替 执行同步块时提高性能。   

主要目的: 在没有多线程竞争的前提下,通过 CAS 减少重量级锁使用操作系统互斥量产生的性能消耗, 说白了先自旋再阻塞 。   

升级时机: 当关闭偏向锁功能或多线程竞争偏向锁会导致偏向锁升级为轻量级锁   

  

假如线程 A 已经拿到锁,这时线程 B 又来抢该对象的锁,由于该对象的锁已经被线程 A 拿到,当前该锁已是偏向锁了。   

而线程 B 在争抢时发现对象头 Mark Word 中的线程 ID 不是线程 B 自己的线程 ID( 而是线程 A) ,那线程 B 就会进行 CAS 操作希望能获得锁。   

此时线程 B 操作中有两种情况:   

如果锁获取成功 ,直接替换 Mark Word 中的线程 ID 为 B 自己的 ID(A → B) ,重新偏向于其他线程 ( 即将偏向锁交给其他线程,相当于当前线程 " 被 " 释放了锁 ) ,该锁会保持偏向锁状态, A 线程 Over , B 线程上位;   

 

如果锁获取失败 ,则偏向锁升级为轻量级锁,此时轻量级锁由原持有偏向锁的线程持有,继续执行其同步代码,而正在竞争的线程 B 会进入自旋等待获得该轻量级锁。   

 

4.5.3、Code演示

如果关闭偏向锁,就可以直接进入轻量级锁  -XX:-UseBiasedLocking

4.5.4、自旋达到一定次数和程度

java6之前

默认启用,默认情况下自旋的次数是 10 次  -XX:PreBlockSpin=10来修改

或者自旋线程数超过cpu核数一半

上述了解即可,别用了。

Java6之后

自适应

自适应意味着自旋的次数不是固定不变的

而是根据: 同一个锁上一次自旋的时间。 拥有锁线程的状态来决定。

4.5.5、轻量锁与偏向锁的区别和不同

争夺轻量级锁失败时,自旋尝试抢占锁

轻量级锁每次退出同步块都需要释放锁,而偏向锁是在竞争发生时才释放锁

4.6、重锁

有大量的线程参与锁的竞争,冲突性很高

锁标志位

 

Code演示

4.7、小总结

各种锁优缺点、synchronized锁升级和实现原理

synchronized锁升级过程总结: 一句话,就是先自旋,不行再阻塞。 

实际上是把之前的悲观锁(重量级锁)变成在一定条件下使用偏向锁以及使用轻量级(自旋锁CAS)的形式 

synchronized在修饰方法和代码块在字节码上实现方式有很大差异,但是内部实现还是基于对象头的MarkWord来实现的。 

JDK1.6之前synchronized使用的是重量级锁, JDK1.6之后进行了优化,拥有了无锁->偏向锁->轻量级锁->重量级锁的升级过程 ,而不是无论什么情况都使用重量级锁。 

偏向锁 :适用于单线程适用的情况,在不存在锁竞争的时候进入同步方法/代码块则使用偏向锁。 

轻量级锁 :适用于竞争较不激烈的情况(这和乐观锁的使用范围类似), 存在竞争时升级为轻量级锁,轻量级锁采用的是自旋锁,如果同步方法/代码块执行时间很短的话,采用轻量级锁虽然会占用cpu资源但是相对比使用重量级锁还是更高效。 

重量级锁 :适用于竞争激烈的情况,如果同步方法/代码块执行时间很长,那么使用轻量级锁自旋带来的性能消耗就比使用重量级锁更严重,这时候就需要升级为重量级锁。 

5、JIT编译器对锁的优化

JIT: Just In Time Compiler,一般翻译为即时编译器

5.1、锁消除

package com.atguigu.juc.lockupgrade;



/**

 * @auther zzyy

 * @create 2021-03-27 15:17

 * 锁消除

 * 从JIT角度看相当于无视它,synchronized (o)不存在了,这个锁对象并没有被共用扩散到其它线程使用,

 * 极端的说就是根本没有加这个锁对象的底层机器码,消除了锁的使用

 */

public class LockClearUPDemo {

    static Object objectLock = new Object();//正常的,有且仅有同一把锁





    public void m1() {

        // 锁消除 ,JIT 会无视它, synchronized( 对象锁 ) 不存在了。不正常的

        Object objectLock = new Object();//锁消除



        synchronized (objectLock) {

            System.out.println("----hello lock");

        }

    }



}

5.2、锁粗化

package com.atguigu.juc.lockupgrade;



import java.util.HashMap;

import java.util.concurrent.locks.Lock;

import java.util.concurrent.locks.ReentrantLock;



/**

 * @auther zzyy

 * @create 2021-03-27 15:19

 * 锁粗化

 * 假如方法中首尾相接,前后相邻的都是同一个锁对象,那JIT编译器就会把这几个synchronized块合并成一个大块,

 * 加粗加大范围,一次申请锁使用即可,避免次次的申请和释放锁,提升了性能

 */

public class LockBigDemo {

    static Object objectLock = new Object();



    public static void main(String[] args) {

        new Thread(() -> {

            synchronized (objectLock) {

                System.out.println("11111");

            }

            synchronized (objectLock) {

                System.out.println("22222");

            }

            synchronized (objectLock) {

                System.out.println("33333");

            }

        }, "a").start();

        new Thread(() -> {

            synchronized (objectLock) {

                System.out.println("44444");

            }

            synchronized (objectLock) {

                System.out.println("55555");

            }

            synchronized (objectLock) {

                System.out.println("66666");

            }

        }, "b").start();

    }

}

  • 0
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值