头图

Summary

In the previous article golang important knowledge: mutex , we introduced the principle implementation of the mutex lock mutex. In addition to mutual exclusion locks in Go, there is also a read-write lock RWMutex, which is mainly used to implement read sharing and write exclusive functions. Today we also analyze the read-write lock by the way to deepen our understanding of Go locks.

The realization principle of read-write lock

The so-called read-write lock actually controls the synchronization and mutual exclusion between Goroutines for the following two scenarios:

  • Multiple goroutines occupy the read lock together, do not affect each other, and can continue the logic code behind themselves.
  • If the write lock is being occupied, the following goroutines will be blocked and wait until the current write lock is released, whether they want to hold a read lock or a write lock.

After clarifying the requirements of the above scenario, the implementation is much simpler. The key is to determine whether it is currently in the write lock state. After all, there needs to be a blocking and waiting action.

According to conventional thinking, we generally use a flag to maintain this state. However, Go official even saves this step.

The number of read locks that must be maintained is used, and when the write lock is occupied, it becomes a negative number.

There are new read and write operations coming in later, you only need to judge whether the value is positive or negative, and a negative number means that the write lock is currently occupied and needs to be blocked waiting.

After the write lock occupancy ends, the value will be restored to a positive number, and new read and write operations can be performed.

RWMutex source code analysis

Next, we go to src/runtime/rwmutex.go to specifically analyze the code structure of RWMutex.

// rwmutex 是一个读写互斥的锁
// 将允许多个 goroutine 持有读锁,但写锁只会有一个持有
// rwmutex 使用了 sync.RWMutex 来辅助写锁互斥
type rwmutex struct {
rLock      mutex    // 用于保护设置 readers, readerPass, writer
readers    muintptr // 休眠等待的 goroutine 读锁队列,等到写锁占有结束后将对应被唤起。
readerPass uint32   // 读锁队列需要跳过的 goroutine 数量,当在写锁结束后会唤起读锁队列里的 goroutine,但有的可能已不在队列里了,这部分需跳过。
wLock  mutex    // 用于 writer 之间的互斥锁
writer muintptr // 等待读完成的 writer

readerCount uint32 // 正在执行读操作的 goroutine数量
readerWait  uint32 // 等待读锁释放的数量。当写锁占有后,前面还有部分读锁在继续着,需要等它们释放才能继续进行。
}

Lock() analysis of RWMutex

func (rw *rwmutex) Lock() {
    // 用于多个写锁之间的的竞争
    lock(&rw.wLock)
    m := getg().m
    // 将读锁数量 readerCount 置为负数,用于判断当前是否处于写锁占有状态,
    // rw.readerCount < 0 则表示当前正在进行写锁占有.
    r := int32(atomic.Xadd(&rw.readerCount, -rwmutexMaxReaders)) + rwmutexMaxReaders
    // 前面还有读锁在进行着,需要等待释放完才能继续
    lock(&rw.rLock)
    if r != 0 && atomic.Xadd(&rw.readerWait, r) != 0 {
        systemstack(func() {
            rw.writer.set(m)
            unlock(&rw.rLock)
            notesleep(&m.park)
            noteclear(&m.park)
        })
    } else {
        unlock(&rw.rLock)
    }
}

RLock() analysis of RWMutex

func (rw *rwmutex) Rlock() {
    acquirem()
    if int32(atomic.Xadd(&rw.readerCount, 1)) < 0 {
        // 读锁数量 readerCount + 1 后小于 0,表示当前正被写锁占有,
        // 等待写锁释放
        systemstack(func() {
            lock(&rw.rLock)
            if rw.readerPass > 0 {
                rw.readerPass -= 1
                unlock(&rw.rLock)
            } else {
                // 等待写锁唤起
                m := getg().m
                m.schedlink = rw.readers
                rw.readers.set(m)
                unlock(&rw.rLock)
                notesleep(&m.park)
                noteclear(&m.park)
            }
        })
    }
}

Analysis of Unlock() of RWMutex

func (rw *rwmutex) Unlock() {
    // 将原来被写锁置为负数的 readerCount 重新恢复回来.
    r := int32(atomic.Xadd(&rw.readerCount, rwmutexMaxReaders))
    if r >= rwmutexMaxReaders {
        throw("unlock of unlocked rwmutex")
    }
    // 唤起之前等待的读锁.
    lock(&rw.rLock)
    for rw.readers.ptr() != nil {
        reader := rw.readers.ptr()
        rw.readers = reader.schedlink
        reader.schedlink.set(nil)
        notewakeup(&reader.park)
        r -= 1
    }
    // 如果 r > 0, 说明读锁队列里有的 goroutine 已不在队列里了,这部分需跳过
    rw.readerPass += uint32(r)
    unlock(&rw.rLock)
    // 解除写锁
    unlock(&rw.wLock)
}

RUnlock() analysis of RWMutex

func (rw *rwmutex) RUnlock() {
    // 如果释放后,readerCount < 0,表示当前写锁正在占有
    if r := int32(atomic.Xadd(&rw.readerCount, -1)); r < 0 {
        if r+1 == 0 || r+1 == -rwmutexMaxReaders {
            throw("runlock of unlocked rwmutex")
        }
        // readerWait == 0,表示前面的读锁都释放完了,
        // 需要唤起写锁
        if atomic.Xadd(&rw.readerWait, -1) == 0 {
            // The last reader unblocks the writer.
            lock(&rw.rLock)
            w := rw.writer.ptr()
            if w != nil {
                notewakeup(&w.park)
            }
            unlock(&rw.rLock)
        }
    }
    releasem(getg().m)
}

Summarize

RWMutex uses the positive or negative of readerCount to determine whether it is currently occupied by a read lock or a write lock.

After the write lock is in the state of holding, the readerCount at this time will be assigned to readerWait, which means that the write lock will be fully occupied after the previous readerWait read locks are released, and the subsequent exclusive operation can be performed.

When the read lock is released, readerWait will be decremented by one until the value is 0, and the write lock can be invoked.

And after the write lock is occupied, even if a new read operation is added, it will not affect the readerWait value, but will only affect the total number of read locks: readerCount.


"read new technology" to follow more push articles.
you can, just like, leave a comment, share, thank you for your support!
Read new technology, read more new knowledge.
阅新技术


lincoln
57 声望12 粉丝

分享有深度、有启示的技术文章;