一般遇到并发问题,都会想到
synchronized
,ReentrantLock
和synchronized
一样都是实现线程同步,但是像比synchronized它更加灵活、强大、增加了轮询、超时、中断等高级功能,可以更加精细化的控制线程同步,它是基于AQS实现的锁,他支持公平锁和非公平锁,同时他也是可重入锁和自旋锁。
目录
一、初识ReentrantLock
ReentrantLock
就是可重入的锁,同一个线程可以多次获得同一把锁
在使用Synchronized
,会存在以下几个问题:
-
不可中断锁,需要线程执行完才会释放锁(synchronized的获取和释放锁由jvm实现)
-
非公平锁
-
Synchronized引入了偏向锁,轻量级锁(自旋锁)后,性能有所提升
Synchronized
-
当线程尝试获取锁的时候,如果获取不到锁会一直阻塞,这个阻塞的过程,用户无法控制
-
如果获取锁的线程进入休眠或者阻塞,除非当前线程异常,否则其他线程尝试获取锁必须一直等待
ReentrantLock
-
可重入锁:可重入锁是指同一个线程可以多次获得同一把锁;ReentrantLock和关键字Synchronized都是可重入锁
-
可中断锁:可中断锁时子线程在获取锁的过程中,是否可以相应线程中断操作。synchronized是不可中断的,ReentrantLock是可中断的
-
公平锁和非公平锁:公平锁是指多个线程尝试获取同一把锁的时候,获取锁的顺序按照线程到达的先后顺序获取,而不是随机插队的方式获取。synchronized是非公平锁,而ReentrantLock是两种都可以实现,不过默认是非公平锁
ReentrantLock
位于java.util.concurrent.locks包下,实现了Lock接口和Serializable接口
public class ReentrantLock implements Lock, java.io.Serializable {……}
synchronized的细粒度和灵活度不够好
ReentrantLock是一把可重入锁
和互斥锁
,它具有与 Synchronized
关键字相同的含有隐式监视器锁(monitor)的基本行为和语义,但是它比 Synchronized
具有更多的方法和功能
二、简单使用
public class ReentrantLockTest {
private static final Lock lock = new ReentrantLock();
public static void test() {
try {
//获取锁
lock.lock();
System.out.println(Thread.currentThread().getName() + "获取到锁了");
//业务代码,使用部分花费100毫秒
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
//释放锁放在finally中。
lock.unlock();
System.out.println(Thread.currentThread().getName() + "释放了锁");
}
}
public static void main(String[] args) {
new Thread(() -> {
test();
}, "线程1").start();
new Thread(() -> {
test();
}, "线程2").start();
}
}
运行结果:
效果和Synchronized的一样,线程1获取到锁了,线程2需要等待线程1释放锁后才可以获取锁