ReadWriteLock:JAVA自带的读写锁

本文通过代码示例探讨了Java中ReentrantLock和ReadWriteLock的使用。在多线程环境下,ReentrantLock无论读写都需要加锁,导致效率降低。而读写锁允许多个线程同时读取,提升了读操作的效率,只有写操作时才会加锁,从而优化了并发性能。实验结果显示,使用读锁进行读操作时,没有上锁,实现了无锁状态,而写锁则确保了写操作的互斥性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.ReentrantLock和synchronized两种锁是不管读操作还是些操作都会进行上锁。但是在一些读场景下,是可以不用上锁的,变量可以同时被多个线程同时读,不会产生脏数据.这时就可以使用到JDK的读写锁。先看ReadWriteLock结构图:这是ReadWriteLock的结构图:

在来看看ReentrantLock的顶级接口Lock的结构:

 这里放两个结构图主要是ReadWriteLock的API可以把锁转换为Lock锁,所以先捋清楚结构。

2.ReentrantLock看看效果:

public class ReadWriteLockTest {
    //排它锁,不管读写场景都加锁
    static Lock reentrantLock  = new ReentrantLock();

    static ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
    //获取读锁
    static Lock readLock = readWriteLock.readLock();
    //获取写锁
    static Lock writeLock = readWriteLock.writeLock();
    //共享变量
    static int value = 0;

    /**
     * 读操作
     * @param lock
     */
    static int read(Lock lock){

        try {
            //上锁
            lock.lock();
            //模拟处理业务
            Thread.sleep(1000);
            return value;
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if(lock != null) lock.unlock();
        }
        return 0;
    }

    /**
     * 写操作
     * @return
     * @Parm newValue:需要修改后的值
     */
    static boolean write(Lock lock, int newValue){

        try {
            //上锁
            lock.lock();
            //模拟处理业务
            Thread.sleep(1000);
            value = newValue;
            return true;
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if(lock != null) lock.unlock();
        }
        return false;
    }

    public static void main(String[] args) throws Exception {
        //到计数器
        CountDownLatch countDownLatch = new CountDownLatch(10);
        long start = System.currentTimeMillis();
        for (int i = 0; i < 10; i++) {
            //让一半线程读,一半线程写
            if(i % 2 == 0){
                new Thread(()->{
                    //读操作
                    read(reentrantLock);
                    countDownLatch.countDown();
                }).start();
            }else {
                new Thread(()->{
                    //读操作
                    write(reentrantLock, 2);
                    countDownLatch.countDown();
                }).start();
            }
        }
        //等待10个线程运行结束才执行,比join()灵活
        countDownLatch.await();
        long end = System.currentTimeMillis();
        System.out.println("reentrantLock:" + (end - start));
    }
}

10个线程读写都要上锁,那么总毫时理论值为10秒左右,结果也严重了我们的猜想。

 2.读锁:

  public static void main(String[] args) throws Exception {
        //到计数器
        CountDownLatch countDownLatch = new CountDownLatch(10);
        long start = System.currentTimeMillis();
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                //读操作
                read(readLock);
                countDownLatch.countDown();
            }).start();
        }
        //等待10个线程运行结束才执行,比join()灵活
        countDownLatch.await();
        long end = System.currentTimeMillis();
        System.out.println("readLock:" + (end - start));
    }

可以发现10个线程同时读是并没有上锁的,也就是无锁状态,多线程情况下时可以提升效率的

3.写锁:

 public static void main(String[] args) throws Exception {
        //到计数器
        CountDownLatch countDownLatch = new CountDownLatch(10);
        long start = System.currentTimeMillis();
        for (int i = 0; i < 10; i++) {
            int newValue = i;
            new Thread(()->{
                //读操作
                write(writeLock, newValue);
                countDownLatch.countDown();
            }).start();
        }
        //等待10个线程运行结束才执行,比join()灵活
        countDownLatch.await();
        long end = System.currentTimeMillis();
        System.out.println("writeLock:" + (end - start));
    }

 可以发现每次写操作都是会上锁的.

总结:在多线程读取数据时情况下并不会上锁,而写锁时会进行上锁的,读场景使用读锁可以提升效率.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值