可重入锁
什么是可重入锁?
当线程获取某个锁后,还可以继续获取它,可以递归调用,而不会发生死锁;
可重入锁案例
程序可重入加锁 A.class,没有发生死锁。
sychronized锁
package com.wnhz.lock.reentrant;
public class SychronizedDemo {
class A {
void a() {
}
}
class B {
void b() {
}
}
public static void main(String[] args) {
System.out.println("准备锁住class A >>>>");
synchronized (A.class) {
System.out.println("锁住class A--> 成功");
System.out.println("准备再次锁住class A >>>>");
synchronized (A.class) {
System.out.println("锁住class A--> 成功");
}
System.out.println("释放class A");
}
}
}
运行结果
准备锁住class A >>>>
锁住class A--> 成功
准备再次锁住class A >>>>
锁住class A--> 成功
释放class A
ReentrantLock
package com.wnhz.lock.reentrant;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class ReentrantLockDemo {
private static final Lock lock = new ReentrantLock();
class A {
void a() {
}
}
class B {
void b() {
}
}
public static void main(String[] args) {
System.out.println("准备锁住class A >>>>");
lock.lock();
try {
System.out.println("锁住class A--> 成功");
System.out.println("准备再次锁住class A >>>>");
lock.lock();
try {
System.out.println("锁住class A--> 成功");
} finally {
System.out.println("释放第二次锁住的class A");
}
} finally {
lock.unlock();
System.out.println("释放第一次锁住的Class A对象");
}
}
}
运行结果
准备锁住class A >>>>
锁住class A--> 成功
准备再次锁住class A >>>>
锁住class A--> 成功
释放第二次锁住的class A
释放第一次锁住的Class A对象
如何保证可重入
当一个线程访问同步块并获取锁时,会在对象头和栈帧中的锁记录里存储偏向的线程ID,以后该线程在进入和退出同步块时不需要进行CAS操作来加锁和解锁,只需简单测试一下对象头的Mark Word里是否存储着指向当前线程的偏向锁。如果测试成功,表示线程已经获得了锁。如果测试失败,则需要再测试一下Mark Word中偏向锁标志是否设置成1:没有则CAS竞争;设置了,则CAS将对象头偏向锁指向当前线程。再维护一个计数器,同个线程进入则自增1,离开再减1,直到为0才能释放。