ReentrantReadWriteLock读写锁及其在 RxCache 中的使用

简介: ReentrantReadWriteLock读写锁及其在 RxCache 中的使用

一. ReentrantReadWriteLock读写锁



Lock 是相当于 synchronized 更面向对象的同步方式,ReentrantLock 是 Lock 的实现。


本文要介绍的 ReentrantReadWriteLock 跟 ReentrantLock 并没有直接的关系,因为它们之间没有继承和实现的关系。


但是 ReentrantReadWriteLock 拥有读锁(ReadLock)和写锁(WriteLock),它们分别都实现了 Lock。

/** Inner class providing readlock */
    private final ReentrantReadWriteLock.ReadLock readerLock;
    /** Inner class providing writelock */
    private final ReentrantReadWriteLock.WriteLock writerLock;


ReentrantReadWriteLock 在使用读锁时,其他线程可以进行读操作,但不可进行写操作。ReentrantReadWriteLock 在使用写锁时,其他线程读、写操作都不可以。


ReentrantReadWriteLock 能够兼顾数据操作的原子性和读写的性能。


1.1 公平锁和非公平锁


从 ReentrantReadWriteLock 的构造函数中可以看出,它默认使用了非公平锁。

/**
     * Creates a new {@code ReentrantReadWriteLock} with
     * default (nonfair) ordering properties.
     */
    public ReentrantReadWriteLock() {
        this(false);
    }
    /**
     * Creates a new {@code ReentrantReadWriteLock} with
     * the given fairness policy.
     *
     * @param fair {@code true} if this lock should use a fair ordering policy
     */
    public ReentrantReadWriteLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
        readerLock = new ReadLock(this);
        writerLock = new WriteLock(this);
    }


在 Java 中所谓公平锁是指,每个线程在获取锁时,会先查看此锁维护的等待队列,如果为队列空或者当前线程线程是等待队列的第一个,则占有锁。否则就会加入到等待队列中,以后按照 FIFO 的顺序从队列中取出。


非公平锁在获取锁时,不会遵循 FIFO 的顺序,而是直接尝试获取锁。如果获取不到锁,则像公平锁一样自动加入到队列的队尾等待。


非公平锁的性能要高于公平锁。


1.2 读锁


读锁是一个共享锁。读锁是 ReentrantReadWriteLock 的内部静态类,它的 lock()、trylock()、unlock() 都是委托 Sync 类实现。


Sync 是真正实现读写锁功能的类,它继承自 AbstractQueuedSynchronizer 。


1.3 写锁


写锁是一个排他锁。写锁也是 ReentrantReadWriteLock 的内部静态类,它的 lock()、trylock()、unlock() 也都是委托 Sync 类实现。写锁的代码类似于读锁,但是在同一时刻写锁是不能被多个线程所获取,它是独占式锁。


写锁可以降级成读锁,下面会介绍锁降级。


1.4 锁降级


锁降级是指先获取写锁,再获取读锁,然后再释放写锁的过程 。锁降级是为了保证数据的可见性。锁降级是 ReentrantReadWriteLock 重要特性之一。


值得注意的是,ReentrantReadWriteLock 并不能实现锁升级。


二. RxCache 中使用读写锁



RxCache 是一款支持 Java 和 Android 的 Local Cache 。目前,支持堆内存、堆外内存(off-heap memory)、磁盘缓存。


github地址:https://github.com/fengzhizi715/RxCache


RxCache 的 CacheRepository 类实现了缓存操作的类,它使用了

ReentrantReadWriteLock 用于保证缓存在读写时避免出现多线程的并发问题。


首先,创建一个读写锁,并获得读锁、写锁的实例。

class CacheRepository {
    private Memory memory;
    private Persistence persistence;
    private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
    private final Lock readLock = lock.readLock();
    private final Lock writeLock = lock.writeLock();
    ......
}


在缓存的读操作时,使用读锁。

boolean containsKey(String key) {
        readLock.lock();
        try {
            if (Preconditions.isBlank(key)) return false;
            return (memory != null && memory.containsKey(key)) || (persistence != null && persistence.containsKey(key));
        } finally {
            readLock.unlock();
        }
    }


在缓存的写操作时,使用写锁。

void remove(String key) {
        writeLock.lock();
        try {
            if (Preconditions.isNotBlank(key)) {
                if (memory != null) {
                    memory.evict(key);
                }
                if (persistence != null) {
                    persistence.evict(key);
                }
            }
        } finally {
            writeLock.unlock();
        }
    }


对于某一个方法,如果在读操作做完之后要进行写操作,则需要先释放读锁,再获取写锁(否则会死锁)。写操作之后,还需要进行读操作的话,可以使用锁降级。

<T> Record<T> get(String key, Type type, CacheStrategy cacheStrategy) {
        readLock.lock();
        try {
            Record<T> record = null;
            if (Preconditions.isNotBlanks(key, type)) {
                switch (cacheStrategy) {
                    case MEMORY: {
                        if (memory!=null) {
                            record = memory.getIfPresent(key);
                        }
                        break;
                    }
                    case PERSISTENCE: {
                        if (persistence!=null) {
                            record = persistence.retrieve(key, type);
                        }
                        break;
                    }
                    case ALL: {
                        if (memory != null) {
                            record = memory.getIfPresent(key);
                        }
                        if (record == null && persistence != null) {
                            record = persistence.retrieve(key, type);
                            if (memory!=null && record!=null && !record.isExpired()) { // 如果 memory 不为空,record 不为空,并且没有过期
                                readLock.unlock(); // 先释放读锁
                                writeLock.lock();  // 再获取写锁
                                try {
                                    if (record.isNeverExpire()) { // record永不过期的话,直接保存不需要计算ttl
                                        memory.put(record.getKey(),record.getData());
                                    } else {
                                        long ttl = record.getExpireTime()- (System.currentTimeMillis() - record.getCreateTime());
                                        memory.put(record.getKey(),record.getData(), ttl);
                                    }
                                    readLock.lock();    // 写锁在没有释放之前,获得读锁 (锁降级)
                                } finally {
                                    writeLock.unlock(); // 释放写锁
                                }
                            }
                        }
                        break;
                    }
                }
            }
            return record;
        } finally {
            readLock.unlock();
        }
    }


三. 总结



ReentrantReadWriteLock 读写锁适用于读多写少的场景,以提高系统的并发性。因此,RxCache 使用读写锁来实现缓存的操作。

相关文章
|
SQL 分布式计算 资源调度
在CDH7.1.1上为Ranger集成OpenLDAP认证
在CDH7.1.1上为Ranger集成OpenLDAP认证
433 0
|
机器学习/深度学习 算法 前端开发
【数据挖掘】袋装、AdaBoost、随机森林算法的讲解及分类实战(超详细 附源码)
【数据挖掘】袋装、AdaBoost、随机森林算法的讲解及分类实战(超详细 附源码)
322 0
|
9月前
|
监控 前端开发 安全
74.8K star!这个开源图标库让界面设计效率提升10倍!
Font Awesome 是全球最受欢迎的图标库和工具包,提供超过2000个免费图标和7000+专业图标,支持网页、桌面应用、移动端等多平台使用。开发者只需几行代码就能为项目添加精美矢量图标,设计师可直接下载SVG进行二次创作。
285 4
|
11月前
|
人工智能 算法 计算机视觉
【01】opencv项目实践第一步opencv是什么-opencv项目实践-opencv完整入门以及项目实践介绍-opencv以土壤和水滴分离的项目实践-人工智能AI项目优雅草卓伊凡
【01】opencv项目实践第一步opencv是什么-opencv项目实践-opencv完整入门以及项目实践介绍-opencv以土壤和水滴分离的项目实践-人工智能AI项目优雅草卓伊凡
407 63
【01】opencv项目实践第一步opencv是什么-opencv项目实践-opencv完整入门以及项目实践介绍-opencv以土壤和水滴分离的项目实践-人工智能AI项目优雅草卓伊凡
|
10月前
|
机器学习/深度学习 人工智能 运维
深度学习在流量监控中的革命性应用
深度学习在流量监控中的革命性应用
385 40
|
8月前
|
人工智能 测试技术 API
PaperBench:OpenAI开源AI智能体评测基准,8316节点精准考核复现能力
PaperBench是OpenAI推出的开源评测框架,通过8316个评分节点系统评估AI智能体复现学术论文的能力,涵盖理论理解、代码实现到实验执行全流程。
595 30
PaperBench:OpenAI开源AI智能体评测基准,8316节点精准考核复现能力
|
前端开发 JavaScript Java
基于Java+Springboot+Vue开发的大学竞赛报名管理系统
基于Java+Springboot+Vue开发的大学竞赛报名管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的大学竞赛报名管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。
715 3
基于Java+Springboot+Vue开发的大学竞赛报名管理系统
|
监控 网络协议 网络性能优化
如何办理支持UDP协议的网络
在当今网络环境中,UDP(用户数据报协议)因传输速度快、延迟低而广泛应用于在线游戏、视频流媒体、VoIP等实时服务。本文详细介绍了办理支持UDP协议网络的方法,包括了解UDP应用场景、选择合适的ISP及网络套餐、购买支持UDP的设备并进行优化设置,以及解决常见问题的策略,帮助用户确保网络稳定性和速度满足实际需求。
|
Unix Linux C语言
在Linux中,grep和egrep命令的区别?
在Linux中,grep和egrep命令的区别?
|
数据采集 存储 前端开发
豆瓣评分9.0!Python3网络爬虫开发实战,堪称教学典范!
今天我们所处的时代是信息化时代,是数据驱动的人工智能时代。在人工智能、物联网时代,万物互联和物理世界的全面数字化使得人工智能可以基于这些数据产生优质的决策,从而对人类的生产生活产生巨大价值。 在这个以数据驱动为特征的时代,数据是最基础的。数据既可以通过研发产品获得,也可以通过爬虫采集公开数据获得,因此爬虫技术在这个快速发展的时代就显得尤为重要,高端爬虫人才的收人也在逐年提高。