ReadWriteLock是Java中的一个接口,提供读/写锁机制,可以用于多线程中。读写锁是一种特殊的锁机制,它可以在某些情况下提高并发性能。在读多写少的情况下,读写锁可以提高并发性能。在读写锁中,允许多个线程同时对共享数据进行读操作,而对共享数据进行写操作时,必须独占锁定。接下来,我们来看一下如何在Java中使用ReadWriteLock接口。
使用ReadWriteLock接口的步骤如下:
-
首先,需要创建一个ReadWriteLock对象,如下所示:
ReadWriteLock rwLock = new ReentrantReadWriteLock();
-
读取操作时,使用读锁lockRead(),如下所示:
rwLock.readLock().lock(); try { // 执行读取操作 } finally { rwLock.readLock().unlock(); }
-
写入操作时,使用写锁lockWrite(),如下所示:
rwLock.writeLock().lock(); try { // 执行写入操作 } finally { rwLock.writeLock().unlock(); }
其中,lock()方法用于获取锁,unlock()方法用于释放锁,必须在finally块中使用,以确保在任何情况下,锁都能够被释放。
注意,如果写入锁已被获取,则所有读取锁和写入锁都将阻塞,直到写入锁被释放。这确保了写入操作不会与读取操作同时进行,避免了数据的不一致性。
下面是一个使用ReadWriteLock接口的示例:
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
public class ReadWriteLockDemo {
private int value = 0;
private ReadWriteLock rwLock = new ReentrantReadWriteLock();
public void read() throws InterruptedException {
System.out.println(Thread.currentThread().getName() + " is trying to read.");
rwLock.readLock().lock();
try {
System.out.println(Thread.currentThread().getName() + " starts reading, value = " + value);
Thread.sleep(1000);
} finally {
rwLock.readLock().unlock();
System.out.println(Thread.currentThread().getName() + " finished reading.");
}
}
public void write(int newValue) throws InterruptedException {
System.out.println(Thread.currentThread().getName() + " is trying to write.");
rwLock.writeLock().lock();
try {
System.out.println(Thread.currentThread().getName() + " starts writing, value = " + newValue);
Thread.sleep(1000);
value = newValue;
} finally {
rwLock.writeLock().unlock();
System.out.println(Thread.currentThread().getName() + " finished writing.");
}
}
public static void main(String[] args) {
ReadWriteLockDemo demo = new ReadWriteLockDemo();
for (int i = 0; i < 10; i++) {
new Thread(() -> {
try {
demo.read();
} catch (InterruptedException e) {
e.printStackTrace();
}
}, "reader-" + i).start();
}
for (int i = 0; i < 2; i++) {
new Thread(() -> {
try {
demo.write((int) (Math.random() * 100));
} catch (InterruptedException e) {
e.printStackTrace();
}
}, "writer-" + i).start();
}
}
}
运行以上代码,可以看到多个读取线程并发读取value的值,同时只有一个写入线程可以修改value的值,避免了数据的不一致性。