likes
comments
collection
share

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

作者站长头像
站长
· 阅读数 37

本文接着上文并发-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源码解读(一)
转载自:https://juejin.cn/post/7245275769220104252
评论
请登录