前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java Concurrent 偏向锁&轻量级锁&重量级锁

Java Concurrent 偏向锁&轻量级锁&重量级锁

作者头像
邹志全
发布2019-07-31 14:51:33
7160
发布2019-07-31 14:51:33
举报
文章被收录于专栏:EffectiveCodingEffectiveCoding

对象头

再说偏向锁之前先来看一下Java 对象头,Java 对象是分为 对象头、实例数据、对齐填充三部分,创建一个Java 对象所消耗和占用的cpu和内存代价都是很高的(尤其是对齐填充这一块,真的会浪费很多内存),和并发相关性最大的是对象头,因为Java 原生锁(sychronized)的信息是存放在Java 对象头中的。如果对象是数组类型,则虚拟机用3个Word(字宽)存储对象头,如果对象是非数组类型,则用2字宽存储对象头。 对象头中的位数依赖于系统的位数: 1、32或64bit存放Mark Word,其中包括存储对象的hashCode或锁信息等。 2、32或64bit存放Class Metadata Address,也就是存储到对象类型数据的指针。 3、如果是数组对象的话,使用32或64bit存放Array length,也就是数组的长度)

mark word中包括HashCode、分代年龄、锁标记位,hashcode用来表示该对象,分代年龄用来JVM分代回收时使用,有兴趣的同学可以看一下我那篇讲垃圾回收算法及垃圾回收器的文章。在运行期间Mark Word里存储的数据会随着锁标志位的变化而变化。Mark Word可能变化为存储以下4种数据:(图片非原创,来自网络)

image.png

在Java SE1.6 以上里锁一共有四种状态,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态,它会随着竞争情况逐渐升级。 注意一点:锁可以升级但不能降级,意味着偏向锁升级成轻量级锁后不能降级成偏向锁。这种锁升级却不能降级的策略,目的是为了提高获得锁和释放锁的效率。(这里和reentrantlock对比下,正好儿是相反的)

偏向锁使用

jvm开启/关闭偏向锁 开启偏向锁:-XX:+UseBiasedLocking -XX:BiasedLockingStartupDelay=0 关闭偏向锁:-XX:-UseBiasedLocking 偏向锁原理:(偏向线程无需每次都要取锁、解锁,就偏向线程来说跟无锁时差别不大) 在实际的应用中经常存在这样一种情况,锁总是一个线程持有,很少发生争用。也就是说锁总是被第一个占用他的线程拥有,这个线程就是锁的偏向线程。那么只需要在锁第一次被拥有的时候,记录下偏向线程ID。这样偏向线程就一直持有着锁,直到竞争发生才释放锁。以后每次同步,检查锁的偏向线程ID与当前线程ID是否一致,如果一致直接进入同步,退出同步也无需每次加锁解锁都去CAS更新对象头,如果不一致意味着发生了竞争,锁已经不是总是偏向于同一个线程了,这时候需要锁膨胀为轻量级锁,才能保证线程间公平竞争锁。

偏向锁加锁过程:

1)访问Mark Word中偏向锁的标识是否设置成1,锁标志位是否为01——确认为可偏向状态。 2)如果为可偏向状态,则测试线程ID是否指向当前线程,如果是,进入步骤(5),否则进入步骤(3)。 3)如果线程ID并未指向当前线程,则通过CAS操作竞争锁。如果竞争成功,则将Mark Word中线程ID设置为当前线程ID,然后执行(5);如果竞争失败,执行(4)。 4)如果CAS获取偏向锁失败,则表示有竞争。当到达全局安全点(safepoint)时获得偏向锁的线程被挂起,偏向锁升级为轻量级锁,然后被阻塞在安全点的线程继续往下执行同步代码。 5)执行同步代码。

轻量级锁

该线程不会阻塞,响应迅速,存在自旋操作,但是会空耗cpu 每次都需要CAS竞争锁,与偏向锁比较来说的话: 1)轻量级锁每次退出同步块都需要释放锁,而偏向锁是在竞争发生时才释放锁 2)每次进入退出同步块都需要CAS更新对象头 3)争夺轻量级锁失败时,自旋尝试抢占锁

加锁过程:

1)在代码进入同步块的时候,如果同步对象锁状态为无锁状态(锁标志位为“01”状态,是否为偏向锁为“0”),JVM首先将在当前线程的栈帧中建立一个名为Lock Record的空间,用于存储锁对象目前的Mark Word的拷贝。 2)拷贝对象头中的Mark Word复制到锁记录中。 3)拷贝成功后,虚拟机将使用CAS操作尝试将对象的Mark Word更新为指向Lock Record的指针,并将Lock record里的owner指针指向object mark word。如果更新成功,则执行步骤(3),否则执行步骤(4)。 4)如果这个更新动作成功了,那么这个线程就拥有了该对象的锁,并且对象Mark Word的锁标志位设置为“00”,即表示此对象处于轻量级锁定状态。 5)如果这个更新操作失败了,虚拟机首先会检查对象的Mark Word是否指向当前线程的栈帧,如果是就说明当前线程已经拥有了这个对象的锁,那就可以直接进入同步块继续执行。否则说明多个线程竞争锁,轻量级锁就要膨胀为重量级锁,锁标志的状态值变为“10”,Mark Word中存储的就是指向重量级锁(互斥量)的指针,后面等待锁的线程也要进入阻塞状态。

重量级锁

吞吐量&长时间执行的代码块,CPU资源较少的 轻量级锁多次竞争失败后,线程阻塞,释放锁后,唤醒阻塞的线程,不使用自旋锁,不会那么消耗CPU,重量级锁适合用在同步块执行时间长的情况下。相较于轻量级来说最大区别应该是不进行自旋操作,直接阻塞,

总结

轻量级锁是为了在线程交替执行同步块时提高性能,而偏向锁则是在只有一个线程执行同步块时进一步提高性能,重量级锁是为处理常存在多线程竞争,同步块执行时间十分长的情况下的锁的方式。

另外还有JVM里面的一点自带优化,具体来说是JIT的优化。 锁粗化(Lock Coarsening):锁粗化的概念应该比较好理解,就是将多次连接在一起的加锁、解锁操作合并为一次,将多个连续的锁扩展成一个范围更大的锁。 锁消除(Lock Elimination):锁消除即删除不必要的加锁操作。根据代码逃逸技术,如果判断到一段代码中,堆上的数据不会逃逸出当前线程,那么可以认为这段代码是线程安全的,不必要加锁。据来说通过运行时JIT编译器的逃逸分析来消除一些没有在当前同步块以外被其他线程共享的数据的锁保护,通过逃逸分析也可以在线程本地Stack上进行对象空间的分配(同时还可以减少Heap上的垃圾收集开销)。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019.07.15 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 对象头
  • 偏向锁使用
    • 偏向锁加锁过程:
    • 轻量级锁
      • 加锁过程:
      • 重量级锁
      • 总结
      相关产品与服务
      对象存储
      对象存储(Cloud Object Storage,COS)是由腾讯云推出的无目录层次结构、无数据格式限制,可容纳海量数据且支持 HTTP/HTTPS 协议访问的分布式存储服务。腾讯云 COS 的存储桶空间无容量上限,无需分区管理,适用于 CDN 数据分发、数据万象处理或大数据计算与分析的数据湖等多种场景。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档