一、摘要
在上篇文章中,我们讲到ReentrantLock
可以保证了只有一个线程能执行加锁的代码。
但是有些时候,这种保护显的有点过头,比如下面这个方法,它仅仅就是只读取数据,不修改数据,它实际上允许多个线程同时调用的。
public class Counter {
private final Lock lock = new ReentrantLock();
private int count;
public int get() {
// 加锁
lock.lock();
try {
return count;
} finally {
// 释放锁
lock.unlock();
}
}
}
站在程序性能的角度,实际上我们想要的是这样的效果。
- 1.读和读之间不互斥,因为只读操作不会有数据安全问题
- 2.写和写之间互斥,避免一个写操作影响另外一个写操作,引发数据计算错误问题
- 3.读和写之间互斥,避免读操作的时候写操作修改了内容,引发数据脏读问题
总结起来就是,允许多个线程同时读,但只要有一个线程在写,其他线程就必须排队等待。
在 JDK 中有一个读写锁ReadWriteLock
,使用它就可以解决这个问题,它可以保证以下两点:
- 1.只允许一个线程写入,其他线程既不能写入也不能读取
- 2.没有写入时,多个线程允许同时读,可以提高程序并发性能
实际上,读写锁ReadWriteLock
里面有两个锁实现,一个是读操作相关的锁,称为共享锁,当多个线程同时操作时,不会让多个线程进行排队等待,大大的提升了程序并发读的执行效率;另一个是写操作相关的锁,称为排他锁,当多个线程同时操作时,只允许一个线程写入,其他线程进入排队等待;两者进行组合操作,就可以实现上面的预期效果。
下面我们一起来看看它的基本用法!
二、ReadWriteLock 基本用法
2.1、读和读共享
读和读之间不互斥,当多个线程进行读的时候,不会让多个线程进行排队等待。
我们可以看一个简单的例子!
public class Counter {
private final ReadWriteLock lock = new ReentrantReadWriteLock();
private int count;
public void read() {
// 加读锁
lock.readLock().lock();
try {
String time = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SSS").format(new Date());
System.out.println(time + " 当前线程:" + Thread.currentThread().getName() + "获得了读锁,count:" + count);
Thread.sleep(5000);
} catch (InterruptedException e) {
e.