synchronized锁拥有的几种状态及表示
在对象头的最后
锁状态 | 对象头的表示 |
---|---|
无锁 | 001 |
偏向锁 | 101 |
轻量锁 | 000 |
重量锁 | 010 |
对于Synchronized在锁的状态上的流程:
1.偏向锁,JVM默认的在启动的时候延迟4秒钟会开启对象的偏向锁,指向第一个访问对象的线程;
2.当有线程对该对象进行加锁并且没有竞争的时候,该对象的偏向锁会被消除,升级膨胀为轻量级锁,
3.当对象有多个线程进行资源竞争的时候,会将轻量级锁升级为重量级锁。
锁的膨胀是不可逆的过程。
但是关于偏向锁,JVM有批量重偏向和批量批量撤销的逻辑。
批量重偏向:当一个线程创建了大量对象并执行了初始的同步操作,后来另一个线程也来将这些对象作为锁对象进行操作,会导偏向锁重偏向的操作。
批量撤销:在多线程竞争剧烈的情况下,使用偏向锁将会降低效率,于是乎产生了批量撤销机制。
在JVM中有关批量重偏向和批量撤销的参数:
intx BiasedLockingBulkRebiasThreshold = 20 默认偏向锁批量重偏向阈值
intx BiasedLockingBulkRevokeThreshold = 40 默认偏向锁批量撤销阈值
锁膨胀、重偏向、撤销Demo
public class LockTest1 {
static List<Node> list = new ArrayList<>();
public static void main(String[] args) throws InterruptedException {
//延迟5秒开启偏向锁
Thread.sleep(5000);
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 100; i++) {
Node node = new Node(UUID.randomUUID().toString(), null);
synchronized (node) {
//锁对应的对象会有偏向锁的产生
list.add(node);
if (i == 18) {
System.out.println("t1:" + ClassLayout.parseInstance(node).toPrintable());
}
}
}
try {