并发-AQS之ReentrantReadWriteLock源码解读(二)

本文接着上文并发-AQS之ReentrantReadWriteLock源码解读(一)继续解读源码

公平锁读写锁加锁解锁

FairSync就两个方法,其他同非公平锁,抽象的非常好
hasQueuedPredecessors(),它检查当前线程的前面是否有等待的线程

static final class FairSync extends Sync {
    private static final long serialVersionUID = -2274990926593161451L;
    final boolean writerShouldBlock() {
        return hasQueuedPredecessors();
    }
    final boolean readerShouldBlock() {
        return hasQueuedPredecessors();
    }
}
// AbstractQueuedSynchronizer
public final boolean hasQueuedPredecessors() {
    Node t = tail; // Read fields in reverse initialization order
    Node h = head;
    Node s;
    return h != t &&
        ((s = h.next) == null || s.thread != Thread.currentThread());
}

读锁ReadLock

lock() ->sync.acquireShared(1) -> tryAcquireShared(arg) -> readerShouldBlock()
lock() ->sync.acquireShared(1) -> tryAcquireShared(arg) -> fullTryAcquireShared(current) -> readerShouldBlock()
其他同非公平锁

写锁WriteLock

lock() ->sync.acquire(1) -> tryAcquire(int arg) -> writerShouldBlock()
其他同非公平锁

其他方法

getReadHoldCount()

方法用于获取当前线程持有的读锁的数量。如果当前线程没有持有读锁,则该方法返回0。
如果当前线程持有一个或多个读锁,则该方法返回持有的读锁数量。
这个方法不会考虑其他线程持有的读锁数量,只返回当前线程持有的读锁数量。

// ReentrantReadWriteLock

public int getReadHoldCount() {
    return sync.getReadHoldCount();
}



// ReentrantReadWriteLock.Sync


final int getReadHoldCount() {
    if (getReadLockCount() == 0)
        return 0;

    Thread current = Thread.currentThread();
    if (firstReader == current)
        return firstReaderHoldCount;

    HoldCounter rh = cachedHoldCounter;
    if (rh != null && rh.tid == getThreadId(current))
        return rh.count;

    int count = readHolds.get().count;
    if (count == 0) readHolds.remove();
    return count;
}
// ReentrantReadWriteLock.Sync


final int getReadLockCount() {
    return sharedCount(getState());
}



getReadLockCount()

用于获取整个锁对象中读锁的数量。
它返回从锁对象创建以来获取的所有读锁的总数,而不仅仅是当前线程持有的读锁数量。
这个方法是用来监控读锁的使用情况,以便在需要时调整并发访问策略。

// ReentrantReadWriteLock.Sync


public int getReadLockCount() {
    return sync.getReadLockCount();
}



getWriteHoldCount()

一个用于获取当前线程持有的写锁数量的方法。
如果当前线程没有持有写锁,则该方法返回0。
如果当前线程持有一个写锁,则该方法返回1。如果当前线程持有多个写锁,则该方法返回持有的写锁数量。

// ReentrantReadWriteLock

public int getWriteHoldCount() {
    return sync.getWriteHoldCount();
}



使用案例

读写锁缓存

    private final Map<String, String> cache = new HashMap<>();
    private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock(false);

    public void put(String key, String value) {
        lock.writeLock().lock();
        try {
            System.out.println("Thread " + Thread.currentThread().getName() + " is writing to cache");
            cache.put(key, value);
            System.out.println("Thread " + Thread.currentThread().getName() + " has written to cache");
        } finally {
            lock.writeLock().unlock();
        }
    }

    public String get(String key) {
        lock.readLock().lock();
        try {
            System.out.println("Thread " + Thread.currentThread().getName() + " is reading from cache");
            return cache.get(key);
        } finally {
            lock.readLock().unlock();
        }
    }

JDK中实现AQS简介

同步工具 与AQS关联 详细介绍
AQS原理讲解 AQS原理介绍 并发-AQS原理讲解
ReentrantLock 使用AQS保存锁重复持有的次数。当一个线程获取锁时,ReentrantLock记录当前获得锁的线程标识,用于检测是否重复获取,以及错误线程试图解锁操作时异常情况的处理。 AQS之Reentrantlonk源码解读
Semaphore 使用AQS同步状态来保存信号量的当前计数。tryRelease会增加计数,acquireShared会减少计数。 Semaphore 源码分析以及AQS共享加解锁
CountDownLatch 在多线程并发执行任务时,有时需要让某些线程等待某些条件达成后再开始执行,这时就可以使用CountDownLatch来实现 CountDownLatch 源码分析
ThreadPoolExecutor 创建线程池中的工作线程worker继承AQS,实现独占资源 参考 并发-AQS之ThreadPoolExecutor源码解读(一)
CyclicBarrier 多个线程等待彼此达到一个共同的屏障点,然后同时继续执行。 并发-AQS之CyclicBarrier源码解读
ReentrantReadWriteLock 可重入读写锁,它允许多个线程同时读取一个共享资源,但只允许一个线程写入该共享资源。 参考 并发-AQS之ReentrantReadWriteLock源码解读(一)

© 版权声明
THE END
喜欢就支持一下吧
点赞0

Warning: mysqli_query(): (HY000/3): Error writing file '/tmp/MYb3Qk4N' (Errcode: 28 - No space left on device) in /www/wwwroot/583.cn/wp-includes/class-wpdb.php on line 2345
admin的头像-五八三
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

图形验证码
取消
昵称代码图片