Java synchronized重量级锁实现过程浅析
一、什么是重量级锁
当有大量的线程都在竞争同一把锁的时候,这个时候加的锁,就是重量级锁。
这个重量级锁其实指的就是JVM内部的ObjectMonitor监视器对象:
ObjectMonitor() { _header = NULL; //锁对象的原始对象头 _count = 0; //抢占当前锁的线程数量 _waiters = 0, //调用wait方法后等待的线程数量 _recursions = 0; //记录锁重入次数 _object = NULL; _owner = NULL; //指向持有ObjectMonitor的线程 _WaitSet = NULL; //处于wait状态的线程队列,等待被唤醒 _WaitSetLock = 0 ; _Responsible = NULL ; _succ = NULL ; _cxq = NULL ; FreeNext = NULL ; _EntryList = NULL ; //等待锁的线程队列 _SpinFreq = 0 ; _SpinClock = 0 ; OwnerIsThread = 0 ; _previous_owner_tid = 0; }
二、重量级锁的演示
public class HightweightLockDemo02 { public static void main(String[] args) { Object objLock = new Object(); new Thread(() -> { synchronized (objLock) { System.out.println(ClassLayout.parseInstance(objLock).toPrintable()); } }, "t1").start(); new Thread(() -> { synchronized (objLock) { System.out.println(ClassLayout.parseInstance(objLock).toPrintable()); } }, "t2").start(); } }
运行程序:
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 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
- Java同步锁Synchronized底层源码和原理剖析(推荐)
- java同步锁的正确使用方法(必看篇)
- 95%的Java程序员人都用不好Synchronized详解
- Java synchronized同步关键字工作原理
- Java synchronized偏向锁的概念与使用
- Java synchronized轻量级锁实现过程浅析
- Java @Transactional与synchronized使用的问题
- Java synchronized与死锁深入探究
- Java synchronized与CAS使用方式详解
- 浅析Java关键词synchronized的使用
- synchronized及JUC显式locks 使用原理解析
- java锁synchronized面试常问总结
- Java HashTable与Collections.synchronizedMap源码深入解析
- Java Synchronized锁的使用详解
- AQS加锁机制Synchronized相似点详解
- Java必会的Synchronized底层原理剖析
- 一个例子带你看懂Java中synchronized关键字到底怎么用
- 详解Java Synchronized的实现原理
- Synchronized 和 ReentrantLock 的实现原理及区别
- Java同步锁synchronized用法的最全总结
相关文章
Java高性能新一代构建工具Maven-mvnd(实践可行版)
这篇文章主要介绍了Java高性能新一代构建工具Maven-mvnd(实践可行版),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2023-06-06java线程池对象ThreadPoolExecutor的深入讲解
在我们的开发中“池”的概念并不罕见,有数据库连接池、线程池、对象池、常量池等等。下面这篇文章主要给大家介绍了关于java线程池对象ThreadPoolExecutor的相关资料,需要的朋友可以参考借鉴,下面来一起看看吧2018-09-09
最新评论