读写锁

本片主要介绍java并发包中的读写锁java.util.concurrent.locks.ReentrantReadWriteLock

传统的锁的劣势

在有些场景中,当有几个线程,其中绝大多数的线程都是对临界区进行读操作,并且读操作不会对临界区的完整性产生破坏,不会修改临界区,此时这些读操作就没有必要还是实行串行的获取锁,读取数据,释放锁 的操作,这样严重影响了效率。

读写锁的优势

读写分离锁可以有效的降低锁竞争,例如线程a,b,c是进行读操作,线程d,e进行写操作,这时如果使用普通的锁,那么线程a,b,c之间是串行操作的,即必须一个一个的挨着来读取数据,但是明明它们一起工作也不会有问题,这就造成了不必要的锁竞争。如果采用读写锁,那么a,b,c之间是可以一起进入临界区进行数据的读取的,仅仅当读写操作,写写操作的时候才会产生一个线程等待另一个线程释放锁的情况发生

例子

public class ReadWriteLock {

    private static Random random = new Random();
    private static Integer value;

    public static void read(Lock lock) throws InterruptedException {
        try {
            lock.lock();
            System.out.println("read value="+value);
            Thread.sleep(1000);
        } finally {
            lock.unlock();
        }
    }

    public static void write(Lock lock) throws InterruptedException{
        try {
            lock.lock();
            value = random.nextInt(100);
            System.out.println("value="+value);
            Thread.sleep(1000);
        } finally {
            lock.unlock();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        final ReentrantLock lock =  new ReentrantLock();
        ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
        final ReadLock readLock = readWriteLock.readLock();
        final WriteLock writeLock = readWriteLock.writeLock();


        Runnable readRunnable = new Runnable() {

            @Override
            public void run() {
                try {
//                  read(lock);
                    read(readLock);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };

        Runnable writeRunnable = new Runnable() {

            @Override
            public void run() {
                try {
//                  write(lock);
                    write(writeLock);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };


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

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

分析,在本程序中分别采用了普通锁(注释的部分放开)和读写锁模拟了对临界区资源读取的过程,发现当采用普通锁的时候,整个程序运行了20多秒,隔一会儿打印一下value(读取),但是采用读写锁后,整个程序只执行了2到3秒,打印(读取)几乎是同时完成的

读写锁的适用场景

读操作次数远远大于写操作次数,那么读写锁就可以发挥最大的功能,提升系统的性能

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redisson的读写锁是一种特殊的锁机制,它允许多个线程同时占有读模式的锁,但一次只能有一个线程占有写模式的锁。当锁处于写模式时,所有试图对该锁加锁的线程都会被阻塞,直到锁被解锁。当锁处于读模式时,所有试图以读模式对其加锁的线程都可以获得访问权,但如果线程希望以写模式对此锁进行加锁,它必须等待所有的线程释放锁。通常情况下,当锁处于读模式锁定状态时,如果有其他线程试图以写模式加锁,读写锁通常会阻塞随后的读模式锁请求,以避免读模式锁长期占用,而等待的写模式锁请求长期阻塞。读写锁适用于对数据结构的读次数远远多于写次数的情况,因为读模式锁定时可以共享,而写模式锁定时是独占的。[3] 在Redisson中,可以通过以下方式使用读写锁: 1. 获取RedissonClient对象。 2. 使用RedissonClient对象的getReadWriteLock方法获取读写锁对象。 3. 使用读写锁对象的readLock方法获取读锁对象,或使用writeLock方法获取写锁对象。 4. 在需要加锁的代码块中,使用读锁对象的lock方法进行加锁,使用写锁对象的lock方法进行加锁。 5. 在加锁的代码块执行完毕后,使用读锁对象的unlock方法进行解锁,使用写锁对象的unlock方法进行解锁。 例如,在验证读锁的请求中,可以使用RedissonClient对象的getReadWriteLock方法获取读写锁对象,然后使用读锁对象的lock方法进行加锁,读取数据后再使用unlock方法进行解锁。在验证写锁的请求中,可以使用相同的方式获取读写锁对象,然后使用写锁对象的lock方法进行加锁,写入数据后再使用unlock方法进行解锁。[2] 总之,Redisson的读写锁是一种灵活且高效的锁机制,可以在多线程环境下实现对共享资源的安全访问。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值