经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Java » 查看文章
Java synchronized重量级锁实现过程浅析
来源:jb51  时间:2023/2/13 8:43:45  对本文有异议

一、什么是重量级锁

当有大量的线程都在竞争同一把锁的时候,这个时候加的锁,就是重量级锁。

这个重量级锁其实指的就是JVM内部的ObjectMonitor监视器对象:

  1. ObjectMonitor() {
  2. _header = NULL; //锁对象的原始对象头
  3. _count = 0; //抢占当前锁的线程数量
  4. _waiters = 0, //调用wait方法后等待的线程数量
  5. _recursions = 0; //记录锁重入次数
  6. _object = NULL;
  7. _owner = NULL; //指向持有ObjectMonitor的线程
  8. _WaitSet = NULL; //处于wait状态的线程队列,等待被唤醒
  9. _WaitSetLock = 0 ;
  10. _Responsible = NULL ;
  11. _succ = NULL ;
  12. _cxq = NULL ;
  13. FreeNext = NULL ;
  14. _EntryList = NULL ; //等待锁的线程队列
  15. _SpinFreq = 0 ;
  16. _SpinClock = 0 ;
  17. OwnerIsThread = 0 ;
  18. _previous_owner_tid = 0;
  19. }

二、重量级锁的演示

  1. public class HightweightLockDemo02 {
  2. public static void main(String[] args) {
  3. Object objLock = new Object();
  4. new Thread(() -> {
  5. synchronized (objLock) {
  6. System.out.println(ClassLayout.parseInstance(objLock).toPrintable());
  7. }
  8. }, "t1").start();
  9. new Thread(() -> {
  10. synchronized (objLock) {
  11. System.out.println(ClassLayout.parseInstance(objLock).toPrintable());
  12. }
  13. }, "t2").start();
  14. }
  15. }

运行程序:

java.lang.Object object internals:
 OFFSET  SIZE   TYPE DESCRIPTION                               VALUE
      0     4        (object header)                           1a 33 c9 e1 (00011010 00110011 11001001 11100001) (-506907878)
      4     4        (object header)                           43 01 00 00 (01000011 00000001 00000000 00000000) (323)
      8     4        (object header)                           e5 01 00 f8 (11100101 00000001 00000000 11111000) (-134217243)
     12     4        (loss due to the next object alignment)
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total
java.lang.Object object internals:
 OFFSET  SIZE   TYPE DESCRIPTION                               VALUE
      0     4        (object header)                           1a 33 c9 e1 (00011010 00110011 11001001 11100001) (-506907878)
      4     4        (object header)                           43 01 00 00 (01000011 00000001 00000000 00000000) (323)
      8     4        (object header)                           e5 01 00 f8 (11100101 00000001 00000000 11111000) (-134217243)
     12     4        (loss due to the next object alignment)
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

可见,当多个线程共同抢占同一把锁的时候,锁对象MarkWord的最后三位是“010”,代表的就是一个重量级锁。

三、重量级锁的原理

以上述代码为例,synchronized获取的锁是重量级锁,synchronized修饰代码块,使用javap -p -v .\HightweightLockDemo02.class指令查看其字节码:

在编译的时候,JVM会在同步块开始位置插入monitorenter指令,在同步块结束位置插入monitorexit指令。当线程执行到monitorenter指令时,会尝试获取对象所对应的Monitor所有权,如果获取成功,则表示获取到了锁,会在Monitor的_owner中存在当前线程的ID,这样它将处于锁定状态,除非退出同步块,否则其他线程无法获取得到这个Monitor。

四、锁的优缺点对比

下表是对各种状态的锁的对比:

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

到此这篇关于Java synchronized重量级锁实现过程浅析的文章就介绍到这了,更多相关Java synchronized 内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号