偏向锁

对一个对象的锁偏向于某个线程,在markword中记录线程id

下次相同的线程来,直接就可以获取锁

轻量级锁

对象的Markword记录锁地址 跟线程栈里面的锁记录Lock Record的锁地址进行交换

重入锁

什么是重入锁 这里举个代码例子

Thread t1 = new Thread(()->{
synchronized(this){
add();
}
}).start(); private synchronized void add(){
xxx;
}

首先你看 线程t1里 synchronized(this)获取了锁,他调用了add方法,但是add方法也需要当前对象的锁吧

正常情况下锁被上面的代码拿了,里面的add是不是不能获取锁,会卡住形成死锁?

这时候重入锁就可以解决这个问题



第一次获取锁的时候 Lock Record里面的锁地址索引跟 Object的MarkWord里面的锁地址进行交换

第二次要获取这个锁 也就是所重入 他发现Object里面的锁地址就是当前线程,于是Thread-0就要重入锁 创一个null的锁记录 代表这次是重入锁

解锁的时候按顺序一个一个解锁 从null开始 到下面那个锁解锁

重量级锁



在轻量级锁的基础上,如果锁膨胀,就会变成重量级锁

首先线程0 Thread-0想要获取对象Object的锁

Thread-0创建自己的锁记录 把锁记录的锁地址跟Object的锁地址进行交换,现在锁记录的锁地址是“无锁”,Object对象里面的锁地址是轻量级锁

现在线程1也想要获取Object的锁 但是注意 现在Object的锁被线程0拿了

线程1看了一眼Object里面的锁地址 怎么是个轻量级锁 怎么不是无锁

这时候 发生了锁的竞争

就需要锁膨胀,膨胀成重量级锁

重量级锁要用到操作系统里面的锁对象 Monitor对象



接下来步骤是这样的

1 Monitor的Owner(锁主人)指向Thread-0 (因为刚刚Object的锁被Thread-0拿了)

2 Thread-1加入到EntryList里面,进行阻塞。

3 等到Thread-0用完了这个锁,把锁示释放开了,Owner指向清空,现在锁没有主人了

4 Thread-0唤醒EntryList里面的阻塞的Thread-1

5 Thread-1获得锁

完成

自旋锁

刚刚的重量级锁,Thread-1是加入到entry list里面去等线程0示范锁,进去就阻塞对吧?

现在可以优化一下,进去entry list先别急着摆烂阻塞,你先试试。

于是Thread1开始自旋,好了没好了没好了没

哦?Thread0好了?那就直接把锁拿到。

为什么这样比较快?因为阻塞会有线程上下文切换,开销很大的

JVM现在很牛。会优化自旋锁,如果前面自旋经常能获取锁,就更愿意让他自旋。如果好几次自旋根本就没用,JVM就会减少自旋次数甚至不自旋,具体的算法我也不懂,反正很智能

批量重偏向

基于偏向锁

这个偏向锁偏向线程1 ,这时候线程1用完了锁,并且以后也不怎么用了。

现在线程2要用这个锁,并且没有线程跟他竞争(注意!没有竞争,有竞争不就膨胀成重量级锁了吗)

这时候要把原来偏向于线程1的锁改掉,改成偏向线程2 这就是锁的重偏向

那么为什么说 批量重偏向,批量是什么意思?

首先批量重偏向是以一个类为单位(一个Class为单位)所有示例对象都算这个Class

比如一个Dog类 有很多实例对象 dog1 dog2 dog3 dog4 dog5

这里有个批量重偏向阈值20

本来所有Dog的示例dog1234567都是偏向线程1的

现在线程2要用这些狗实例,慢慢的dog1从偏向线程1变成偏向线程2,dog2也偏向线程2,dog3也变,dog4也变

如此变了20次(阈值)的时候,jvm觉得,咋回事,偏向锁的出现本来不是为了加快效率吗,你这样一直变变变,不是反而慢了吗

ok,既然这些狗对象一直变成偏向线程2,那就统统给我偏向到线程2。比如你创建了40条狗,二十条狗被你从偏向线程1改成偏向线程二,就触发了批量重偏向,现在所有的狗都偏向于线程二了

批量撤销

ok 批量冲偏向已经很好用了吧 现在我再来个新阈值 叫做批量撤销阈值:40 刚刚触发20次的时候会触发批量重偏向,让所有的狗都去线程二,偏向锁还是可以用的。

现在撤销了四十次,JVM就觉得,搞毛,一直撤销,都四十次了,这样效率很低。化身恶魔,都别用。

40次的撤销触发了批量撤销后,所有的DOG的实例(dog1,dog2,dog345678)统统变成不可偏向!

不仅以前创建的变成不可偏向,新创建的小狗实例也不准用偏向

OK 这就是我对java里面的锁的理解,希望能帮到大家,如果有错欢迎指出一起讨论!

大白话讲Java的锁的更多相关文章

  1. java的锁机制

    一段synchronized的代码被一个线程执行之前,他要先拿到执行这段代码的权限,在Java里边就是拿到某个同步对象的锁(一个对象只有一把锁): 如果这个时候同步对象的锁被其他线程拿走了,他(这个线 ...

  2. Java偏向锁实现原理(Biased Locking)

    http://kenwublog.com/theory-of-java-biased-locking 阅读本文的读者,需要对Java轻量级锁有一定的了解,知道lock record, mark wor ...

  3. java的锁机制——synchronized

    一段synchronized的代码被一个线程执行之前,他要先拿到执行这段代码的权限,在java里边就是拿到某个同步对象的锁(一个对象只有一把锁): 如果这个时候同步对象的锁被其他线程拿走了,他(这个线 ...

  4. 从零讲Java,给你一条清晰地学习道路!该学什么就学什么!

                                             从零讲JAVA ,给你一条 清晰地学习道路!该学什么就学什么! 1.计算机基础: 1.1数据机构基础: 主要学习:1. ...

  5. JAVA中锁的解决方案

    前言 在上一节中,我们给大家介绍了什么是锁,以及锁的使用场景,我相信大家对锁的定义,以及锁的重要性都有了比较清晰的认识.在这一节中,我们会给大家继续做深入的介绍,介绍JAVA为我们提供的不同种类的锁. ...

  6. java 分布式锁方案

    第一步,自身的业务场景: 在我日常做的项目中,目前涉及了以下这些业务场景: 场景一: 比如分配任务场景.在这个场景中,由于是公司的业务后台系统,主要是用于审核人员的审核工作,并发量并不是很高,而且任务 ...

  7. JAVA线程锁-读写锁

    JAVA线程锁,除Lock的传统锁,又有两种特殊锁,叫读写锁ReadWriteLock 其中多个读锁不互斥,读锁和写锁互斥,写锁和写锁互斥 例子: /** * java线程锁分为读写锁 ReadWri ...

  8. Java线程锁一个简单Lock

    /** * @author * * Lock 是java.util.concurrent.locks下提供的java线程锁,作用跟synchronized类似, * 单是比它更加面向对象,两个线程执行 ...

  9. paip.提升性能----java 无锁结构(CAS, Atomic, Threadlocal, volatile, 函数式编码, 不变对象)

    paip.提升性能----java 无锁结构(CAS, Atomic, Threadlocal, volatile, 函数式编码, 不变对象) 1     锁的缺点 2     CAS(Compare ...

随机推荐

  1. 左手Cookie“小甜饼”,右手Web Storage

    目录 1. Web Storage 2. Cookie机制 3. 二者的联系与区别 1.Web Storage 1.1 概述 Web Storage是HTML5提供的一种新的浏览器端数据储存机制,它提 ...

  2. oracle 多列求和

    第一种: select sum(decode(count1,null,0,count1) +decode(count2,null,0,count2) +decode(count3,null,0,cou ...

  3. docker 国内镜像源

    参考:docker 国内镜像源 作者:Joncc Linux环境 # vi /etc/docker/daemon.json { "registry-mirrors": [" ...

  4. numpy教程04---ndarray的索引

    欢迎关注公众号[Python开发实战], 获取更多内容! 工具-numpy numpy是使用Python进行数据科学的基础库.numpy以一个强大的N维数组对象为中心,它还包含有用的线性代数,傅里叶变 ...

  5. 变量 数据类型 条件if语句

    python是解释型  弱类型编程语言;  "优雅", "明确", "简单";  开发效率非常高;  可移植性;  可扩展性; 可嵌入型. ...

  6. python---变量、常量、注释、基本数据类型

    变量 变量:将运算的中间结果暂存到内存中,以便后续程序调用. 变量的命令规则: 变量由字母.数字.下划线组合而成. 不可以数字开头,更不能全是数字. 不能是python的关键字. 不要用中文. 名字要 ...

  7. AgileConfig 1.6.0 发布 - 支持服务注册与发现

    大家好,好久没有输出博文了,一是因为比较忙,另外一个原因是最近主要的精力是在给 AgileConfig 添加一个新的功能:服务注册与发现. 先说说为什么会添加这个功能.我自己的项目是用 Consul ...

  8. Servlet 标准下载地址

    Servlet 标准下载地址 java 技术标准开发社区 https://jcp.org/ 3.0 https://jcp.org/aboutJava/communityprocess/mrel/js ...

  9. Java学习day26

    进程.多任务 1.例如吃饭的时候玩手机,边上厕所边玩手机,看似是同时做多个事情,本质上我们的大脑在同一时间只做了一件事情,这就是多任务 2.道路窄的时候容易造成拥堵,可以拓宽道路,加多车道,同一个方向 ...

  10. HCIE笔记-第七节-ICMP+ARP

    ICMP重定向 作用:解决网络中的次优路径 触发:当某一个设备收到一个数据,进行转发时发现还要从该接口进行转发,于是触发ICMP重定向. 报文:Type=5,Code=0 ARP -- 地址解析协议 ...