一、显式锁简介
显式锁,这个叫法是相对于隐式锁synchronized而言的,加锁和解锁都要用户显式地控制。显示锁Lock是在Java5中添加到jdk的,同synchronized一样,这也是一种协调共享对象访问的机制。但是它不是用来替代内置锁的,而是一种可选择的高级功能。
1、Lock接口提供了synchronized关键字不具备的主要特性:
- 尝试非阻塞获取锁:当前线程尝试获取锁,如果这一时刻,锁没有被其他线程占有,那么成功获取锁并返回。
- 能被中断地获取锁:当线程正在等待获取锁,则这个线程能够 响应中断,即当中断来了,线程不会阻塞等待获取锁,抛出中断异常。
- 超时获取锁:在指定的截止时间前获取锁,如果截止时间到了仍旧无法获取锁,则返回;
关于Lock与synchronized的区别,请参考我的上一篇博文。
2、两种显式锁
JDK中提供了两种显式锁,即Lock的实现方式有两种:ReentrentLock(重入锁)、ReentrantReadWriteLock.ReadLock 和 ReentrantReadWriteLock.WriteLock(这两个锁是由其父类ReentrantReadWriteLock 控制使用,可视为一体,称为读写锁)。具体的Lock接口的继承结构,可参考下图:
二、Lock接口的API
方法名称 | 描述 |
---|---|
void lock( ) | 阻塞地获取锁,直到获取到锁才返回,而且是不可中断的。 |
void lockInterruptibly( ) throws InterruptedException | 可中断地获取锁,与lock()方法的不同之处,在于该方法在阻塞等待锁的过程中会 响应中断。 |
boolean tryLock( ) | 尝试非阻塞地获取锁,即调用该方法后,立刻返回,成功获取锁,返回true,失败则返回false。 |
boolean tryLock(long time, TimeUnit unit) throws InterruptedException | 可中断的超时获取锁,在以下3种情况下会返回: 1. 当前线程在指定时间内获得了锁; 2. 当前线程在指定时间内被中断; 3. 指定时间结束(超时结束),返回false; |
void unlock( ) | 释放锁。 |
Condition newCondition() | 等待通知组件,当前线程只有获得了锁,才能调用该组件的wait()方法,调用后,线程将会释放锁 |
三、ReentrentLock重入锁 详解
ReentrentLock 是Lock接口的常用的实现类,是一个可重入的互斥锁 Lock,它具有与使用 synchronized 方法和语句所访问的隐式监视器锁相同的一些基本行为和语义,但功能更强大。更加灵活(实现了显示锁的特性)。下面将由ReentrentLock 来介绍体验显示锁的特性:
1、可中断锁与不可中断锁 - - lockInterruptibly( )、lock( )
Lock接口不仅提供了不可中断锁(synchronized是不可中断的),还有可中断锁。在某些应用场景下,可中断锁的用处很大:当检测到线程等待锁的时间过长,不能继续等待,需要进行下一步操作;或者某个任务已经完成了,则中断其他等待锁来完成这个任务的线程。
显式锁的加锁和解锁都是由用户来操作,所以用户一旦忘记释放锁了,很可能就会造成线程讥饿。正确的用是使用 try-finally 确保锁能被正确释放。
public void m() {
lock.lock(); // block until condition holds
try {
// ... method body
} finally {
lock.unlock()
}
}
如果是可中断方式获取锁 lockInterruptibly,则要 try-finally 要处于 捕获中断异常的 try-catch 块间,或者在方法上抛出中断异常。
public void method() throws InterruptedException {
//抛出中断异常
lock.lockInterruptibly();
try {
//方法体.....
}
finally {
lock.unlock();
}
}
public void m() {
try {
lock.lockInterruptibly();
try{
// ... method body
}finally{
lock.unlock();
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
中断锁的例子:
//静态变量
static Lock lock =new ReentrantLock();
public static void