Java锁相关(二)
前言
- 上一章说到“若CAS抢锁失败则进入锁升级,若复制Hashcode Age 0 失败也会进入锁升级”,所说的锁升级是由轻量级锁(Light-weight locked)升级为重量级锁(Heavy-weight locked)的一个过程。这其中和重量级锁对应的内存信息就是Monitor Address也是本章所展示的内容。
对象监视器(Monitor)
-
java当中每个对象都会有一个对象监视器,对象监视器有很大部分的作用是用来实现锁的机制(Synchronized)。
-
当线程自旋过多时则进入锁池EntryList,此时线程的状态为Blocked阻塞状态。
-
在Synchronized同步关键字当中抢到锁才能使用Wait、Notify;其中Wait语义为:释放owner当中的线程,使其进入Waitset里面;线程进入waiting状态。
-
owner为null表示未被占用,阻塞的线程可以去抢锁;由于Entrylist并不能保证是公平的,所以当Entrylist中的线程(T2)去抢锁时,其它线程(T4)同时也可以来抢锁,而且有可能会抢锁成功,所以说Synchronized同步关键字是非公平的锁。
-
T1线程执行Notify后唤醒该线程,然后使其去抢锁;这时发现owner被占用了,线程马上进入Entrylist锁池里面来,并在后面排队不会因为是从等待池唤醒的线程而有优先权。
-
T2线程代码块执行完,T2线程会执行monitory exit,锁就会退出;T2就会从monitor中出来
-
Waitset对应线程的waiting状态,调用notify后线程进入blocked状态。
-
Synchronized同步关键字会使线程先去抢锁,线程失败后会进入锁池排队,所以说Synchronized是悲观锁。
偏向锁
-
若锁不需要争抢,连CAS机制都用不到,这时就会使用偏向锁。偏向锁即使使用完了也不会去解锁,也就是说不会释放锁。下一次若判断Thread ID 是自己就直接用,根本不需要解释。
-
若没有其它线程和T1线程争抢锁,T1一直会处于偏向锁状态,不会释放锁;当有其它线程(T2)过来争抢,就会进行锁升级,由偏向锁状态升级为轻量级锁状态(lock record address 00),抢锁失败的线程就会进入自旋,自旋的线程会进入锁池,如果自旋达到一定的次数就会进入重量级锁。
-
若偏向锁的线程都已经释放了锁(所释放),会进入未锁定 关闭偏向锁状态(Hash code age 0 01),下次抢锁不再经过轻量级锁,直接进入重量级锁。