可重入锁,Java自己实现的锁,继承了Lock类,有两种构造。
ReentrantLock在同一个时间点只能被一个线程获取(当某线程获取到“锁”时,其它线程就必须等待);ReentraantLock是通过一个FIFO的等待队列来管理获取该锁所有线程的。在“公平锁”的机制下,线程依次排队获取锁;而“非公平锁”在锁是可获取状态时,不管自己是不是在队列的开头都会获取锁。
相同: ReentrantLock提供了synchronized类似的功能和内存语义。
不同
[1]与synchronized相比,ReentrantLock提供了更多,更加全面的功能,具备更强的扩展性。例如:时间锁等候,可中断锁等候,锁投票。
[2]ReentrantLock还提供了条件Condition,对线程的等待、唤醒操作更加详细和灵活,所以在多个条件变量和高度竞争锁的地方,ReentrantLock更加适合(下面会阐述Condition)。
[3]ReentrantLock提供了可轮询的锁请求。它会尝试着去获取锁,如果成功则继续,否则可以等到下次运行时处理,而synchronized则一旦进入锁请求要么成功,要么一直阻塞,所以相比synchronized而言,ReentrantLock会不容易产生死锁些。
[4]ReentrantLock支持更加灵活的同步代码块,但是使用synchronized时,只能在同一个synchronized块结构中获取和释放。注:ReentrantLock的锁释放一定要在finally中处理,否则可能会产生严重的后果。
[5]ReentrantLock支持中断处理,且性能较synchronized会好些。
有一个与锁相关的获取计数器,如果拥有锁的某个线程再次得到锁,那么获取计数器就加1,然后锁需要被释放两次才能获得真正释放。
通过构造方法实现,默认非公平锁。且非公平锁性能高于公平锁性能。
如果获取一个锁是按照请求的顺序得到的,那么就是公平锁,否则就是非公平锁,允许线程插队。
公平锁表示线程获取锁的顺序是按照线程加锁的顺序来分配的,即先来先得的FIFO先进先出顺序。而非公平锁就是一种获取锁的抢占机制,是随机获得锁的,和公平锁不一样的就是先来的不一定先得到锁,这个方式可能造成某些线程一直拿不到锁,结果也就是不公平的了。
// 简单示例
public class ReentrantLockThread implements Runnable {
ReentrantLock lock = new ReentrantLock();
@Override
public void run() {
lock.lock();
try {
for (int i = 0; i < 3; i++) {
System.out.println(Thread.currentThread().getName() + "输出:" + i);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public static void main(String[] args) {
ReentrantLockThread thread1 = new ReentrantLockThread();
new Thread(thread1, "a").start();
new Thread(thread1, "b").start();
}
}
// 锁的重入
public class TestRetryLock {
ReentrantLock lock = null;
public TestRetryLock() {
lock = new ReentrantLock();
}
public static void main(String[] args) {
TestRetryLock retryLock = new TestRetryLock();
try {
retryLock.testReentry(1);
// 能执行到这里而不阻塞,表示锁可重入
retryLock.testReentry(2);
retryLock.testReentry(3);
} finally {
// 释放重入测试的锁,要按重入的数量解锁,否则其他线程无法获取该锁。
retryLock.getLock().unlock();
retryLock.getLock().unlock();
retryLock.getLock().unlock();
}
}
public ReentrantLock getLock() {
return lock;
}
public void testReentry(int count) {
lock.lock();
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
System.out.println(sdf.format(new Date()) + " " + Thread.currentThread().getName()
+ " get lock. " + count);
}
}
// 公平和非公平锁
public class LockService {
private ReentrantLock lock;
public LockService(boolean fair){
lock = new ReentrantLock(fair);
}
public void serviceMethod() {
lock.lock();
try {
System.out.println("ThreadName=" + Thread.currentThread().getName() + " 获得锁定");
} finally {
lock.unlock();
}
}
public static void main(String[] args) {
LockService lockService = new LockService(true);
Runnable runnable = () -> {
System.out.println("**线程: " + Thread.currentThread().getName() + " 运行了 ");
lockService.serviceMethod();
};
for (int i = 0; i < 10; i++) {
new Thread(runnable).start();
}
}
}
Condition类能实现synchronized和wait、notify搭配的功能,另外比后者更灵活,Condition可以实现多路通知功能,也就是在一个Lock对象里可以创建多个Condition(即对象监视器)实例,线程对象可以注册在指定的Condition中,从而可以有选择的进行线程通知,在调度线程上更加灵活。而synchronized就相当于整个Lock对象中只有一个单一的Condition对象,所有的线程都注册在这个对象上。线程开始notifyAll时,需要通知所有的WAITING线程,没有选择权,会有相当大的效率问题。
Condition是个接口,基本的方法就是await()和signal()方法。
Condition依赖于Lock接口,生成一个Condition的基本代码是lock.newCondition()
调用Condition的await()和signal()方法,都必须在lock保护之内,就是说必须在lock.lock()和lock.unlock之间才可以使用。
Conditon中的await()对应Object的wait(),Condition中的signal()对应Object的notify(),Condition中的signalAll()对应Object的notifyAll()。
condition是跟着线程来的,即一个线程可以拥有一个condition,而不是一个ReentrantLock拥有一个condition
// 简单示例
public class ConditionService {
// 实例化一个ReentrantLock对象
private ReentrantLock lock = new ReentrantLock();
// 为线程A注册一个Condition
public Condition conditionA = lock.newCondition();
// 为线程B注册一个Condition
public Condition conditionB = lock.newCondition();
public void awaitA() {
lock.lock();
try {
System.out.println(Thread.currentThread().getName() + "进入了awaitA方法");
long timeBefore = System.currentTimeMillis();
// 执行conditionA等待
conditionA.await();
long timeAfter = System.currentTimeMillis();
System.out.println(Thread.currentThread().getName() + "被唤醒");
System.out.println(Thread.currentThread().getName() + "等待了: " + (timeAfter - timeBefore) / 1000 + "s");
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public void awaitB() {
lock.lock();
try {
System.out.println(Thread.currentThread().getName() + "进入了awaitB方法");
long timeBefore = System.currentTimeMillis();
// 执行conditionA等待
conditionB.await();
long timeAfter = System.currentTimeMillis();
System.out.println(Thread.currentThread().getName() + "被唤醒");
System.out.println(Thread.currentThread().getName() + "等待了: " + (timeAfter - timeBefore) / 1000 + "s");
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public void signallA() {
lock.lock();
try {
System.out.println("启动唤醒程序");
// 唤醒所有注册conditionA的线程
conditionA.signalAll();
} finally {
lock.unlock();
}
}
public void signallB() {
lock.lock();
try {
System.out.println("启动唤醒程序");
// 唤醒所有注册conditionB的线程
conditionB.signalAll();
} finally {
lock.unlock();
}
}
}
public static void main(String[] args) throws InterruptedException {
ConditionService conditionService = new ConditionService();
new Thread(() -> conditionService.awaitA(), "A").start();
new Thread(() -> conditionService.awaitB(), "B").start();
Thread.sleep(2000);
// 唤醒持有ConditionA的线程
conditionService.signallA();
Thread.sleep(2000);
// 唤醒持有ConditionB的线程
conditionService.signallB();
}
// 生产者消费者模式
public class SomethingFactory {
/**
* 工厂最大库存容量
*/
public static final int MAX_SIZE = 10;
private int currentNumber = 0;
private ReentrantLock lock = new ReentrantLock();
private Condition condition = lock.newCondition();
public void produce(int needNum) {
lock.lock();
try {
while (currentNumber + needNum > MAX_SIZE) {
System.out.println("要生产的产品数量" + needNum + "超过剩余库存量" + (MAX_SIZE - currentNumber) + ",暂时不能执行生产任务!");
condition.await();
}
System.out.println(Thread.currentThread().getName() + "---生产商品---");
currentNumber += needNum;
System.out.println("已经生产了" + needNum + "个产品,现仓储量为" + currentNumber);
// 通知消费者
condition.signalAll();
} catch (Exception e) {
} finally {
lock.unlock();
}
}
public void consume(int needNum) {
lock.lock();
try {
while (currentNumber < needNum) {
System.out.println("要消费的产品数量" + needNum + "超过剩余库存量" + (currentNumber) + ",暂时不能执行消费任务!");
condition.await();
}
System.out.println(Thread.currentThread().getName() + "---消费商品---");
currentNumber -= needNum;
System.out.println("已经消费了" + needNum + "个产品,现仓储量为" + currentNumber);
// 提醒生产者生产
condition.signalAll();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}
public static void main(String[] args) {
SomethingFactory factory = new SomethingFactory();
new Thread(() -> {
int count = 0;
while (count++ < 5) {
factory.consume(1);
}
}, "consumer-A").start();
new Thread(() -> {
int count = 0;
while (count++ < 5) {
factory.produce(1);
}
}, "producer").start();
new Thread(() -> {
int count = 0;
while (count++ < 5) {
factory.consume(1);
}
}, "consumer-B").start();
}