在java中我们听过或者用过的锁有很多种:公平锁/非公平锁、可重入锁/不可重入锁、共享锁/排他锁、乐观锁/悲观锁、偏向锁/轻量级锁/重量级锁。其实这些都是在不同维度或者锁优化角度对锁的一种叫法,我们在程序中用到的也就那么几种,比如synchronized,ReentrantLock,ReentrantReadWriteLock等
这里着重说一下可重入锁
什么是可重入锁?
一个锁对象被当前线程持有了,当前线程可以继续获取该锁对象。
我们常用的synchronized锁 和 ReentrantLock锁都是可重入锁
public class Test01 {
public static void main(String[] args) {
synchronized(args) {
synchronized(args) {
System.out.println("123456789");
}
}
Lock lock = new ReentrantLock();
lock.lock();
lock.lock();
System.out.println("123456789");
lock.unlock();
lock.unlock();
该段代码为可重入锁的的基本演示。
为什么有可重入锁?
类似于a方法调用b方法,就需要可重入锁的设计
public static synchronized void a() {
b();
}
public static synchronized void b() {}
}
可重入锁的使用场景?
可重入锁主要用在线程需要多次进入临界区代码时,需要使用可重入锁
那么,可重入锁的实现原理是什么呢?
每一个锁关联一个线程持有者和计数器,当计数器为 0 时表示该锁没有被任何线程持有,那么任何线程都可能获得该锁而调用相应的方法;当某一线程请求成功后,JVM会记下锁的持有线程,并且将计数器置为 1;此时其它线程请求该锁,则必须等待;而该持有锁的线程如果再次请求这个锁,就可以再次拿到这个锁,同时计数器会递增;当线程退出同步代码块时,计数器会递减,如果计数器为 0,则释放该锁。