Java的锁—读写锁(WriteReadLock)

对某一数据进行线程安全的读写操作,那么就要利用锁来进行线程同步,习惯的做法是,读的时候加锁,禁止其他的读写操作,写的时候,更是要对其他读写操作禁止。那么读写锁就是要更细粒度的进行锁的操作。读的时候,允许其他的线程的读操作,禁止其他线程的写操作等。那么具体的总结如下表:

-
非阻塞阻塞
阻塞阻塞

比较极端的例子,如果系统中写的次数远远小于读的次数,那么这更能突出读写锁的优势:

public class WriteReadLockDemo {

    private static Lock lock = new ReentrantLock();
    private static ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
    private static Lock readLock = readWriteLock.readLock();
    private static Lock writeLock = readWriteLock.writeLock();

    private int value;

    public Object handleRead(Lock lock) throws InterruptedException {
        try {
            lock.lock();
            Thread.sleep(1000);
            return value;
        } finally {
            lock.unlock();
        }
    }

    public void handdleWrite(Lock lock, int index) throws InterruptedException {
        try {
            lock.lock();
            Thread.sleep(1000);
            value = index;
        } finally {
            lock.unlock();
        }
    }

    public static void main(String[] args) {
        WriteReadLockDemo writeReadLockDemo = new WriteReadLockDemo();
        Runnable readRunnable = () -> {
            try {
                // 下面两行是普通锁和读写锁
// writeReadLockDemo.handleRead(readLock);
                    writeReadLockDemo.handleRead(lock);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        };

        Runnable writeRunnable = () -> {
            try {
                // 下面两行是普通锁和读写锁
// writeReadLockDemo.handdleWrite(writeLock, new Random().nextInt());
                    writeReadLockDemo.handdleWrite(lock, new Random().nextInt());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        };

        for (int i = 0; i < 18; i++) {
            new Thread(readRunnable).start();
        }

        for (int i = 18; i < 20; i++) {
            new Thread(writeRunnable).start();
        }

    }

}

假设读写操作都很耗时,每个读或写操作都需要1秒。

main方法里面,两个for循环,分别启动18个读线程,2个写线程。程序的读方法,写方法分别有一个注释,是表示用普通锁和读写锁,用读写锁2秒执行完成,而利用普通锁则需要20秒。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中,读写锁是通过`ReentrantReadWriteLock`类实现的。这个类提供了两个,一个读和一个写,可以通过它们来实现对共享数据的读写操作。 下面是一个简单的示例代码,演示了如何使用读写锁来实现对共享数据的读写操作。 ```java import java.util.concurrent.locks.ReentrantReadWriteLock; public class ReadWriteLockDemo { private static final ReentrantReadWriteLock lock = new ReentrantReadWriteLock(); private static int count = 0; public static void main(String[] args) { new Thread(() -> { while (true) { try { lock.writeLock().lock(); count++; System.out.println(Thread.currentThread().getName() + " write count = " + count); Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.writeLock().unlock(); } } }, "write-thread").start(); new Thread(() -> { while (true) { try { lock.readLock().lock(); System.out.println(Thread.currentThread().getName() + " read count = " + count); Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.readLock().unlock(); } } }, "read-thread").start(); } } ``` 在这个例子中,我们创建了一个`ReentrantReadWriteLock`对象作为共享,并定义了一个共享变量`count`。然后,我们创建了两个线程,一个写线程和一个读线程。写线程每隔一秒钟增加一次`count`的值,并打印出来,而读线程每隔一秒钟读取`count`的值,并打印出来。 在写线程中,我们使用`lock.writeLock().lock()`获取写,并在使用完共享变量后,使用`lock.writeLock().unlock()`释放写。而在读线程中,我们使用`lock.readLock().lock()`获取读,并在使用完共享变量后,使用`lock.readLock().unlock()`释放读。这样就可以保证多个读线程可以同时读取共享变量,而写线程需要独占。 需要注意的是,写是独占,在写被持有期间,其他线程无法获取写和读,而读是共享,在读被持有期间,其他线程可以同时获取读,但不能获取写

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值