欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang?Mutex錯(cuò)過會(huì)后悔的重要知識(shí)點(diǎn)分享

 更新時(shí)間:2023年07月26日 10:18:13   作者:碼一行  
互斥鎖?Mutex?是并發(fā)控制的一個(gè)基本手段,是為了避免并發(fā)競爭建立的并發(fā)控制機(jī)制,本文主要為大家整理了一些Mutex的相關(guān)知識(shí)點(diǎn),希望對(duì)大家有所幫助

Go Mutex 的基本用法

Mutex 我們一般只會(huì)用到它的兩個(gè)方法:

  • Lock:獲取互斥鎖。(只會(huì)有一個(gè)協(xié)程可以獲取到鎖,通常用在臨界區(qū)開始的地方。)
  • Unlock: 釋放互斥鎖。(釋放獲取到的鎖,通常用在臨界區(qū)結(jié)束的地方。)

Mutex 的模型可以用下圖表示:

說明:

  • 同一時(shí)刻只能有一個(gè)協(xié)程獲取到 Mutex 的使用權(quán),其他協(xié)程需要排隊(duì)等待(也就是上圖的 G1->G2->Gn)。
  • 擁有鎖的協(xié)程從臨界區(qū)退出的時(shí)候需要使用 Unlock 來釋放鎖,這個(gè)時(shí)候等待隊(duì)列的下一個(gè)協(xié)程可以獲取到鎖(實(shí)際實(shí)現(xiàn)比這里說的復(fù)雜很多,后面會(huì)細(xì)說),從而進(jìn)入臨界區(qū)。
  • 等待的協(xié)程會(huì)在 Lock 調(diào)用處阻塞,Unlock 的時(shí)候會(huì)使得一個(gè)等待的協(xié)程解除阻塞的狀態(tài),得以繼續(xù)執(zhí)行。

這幾點(diǎn)也是 Mutex 的基本原理。

Go Mutex 原子操作

Mutex結(jié)構(gòu)體定義:

type Mutex struct {
   state int32 // 狀態(tài)字段
   sema  uint32 // 信號(hào)量
}

其中 state 字段記錄了四種不同的信息:

這四種不同信息在源碼中定義了不同的常量:

const (
   mutexLocked      = 1 << iota // 表示有 goroutine 擁有鎖
   mutexWoken                   // 喚醒(就是第 2 位)
   mutexStarving                // 饑餓(第 3 位)
   mutexWaiterShift = iota      // 表示第 4 位開始,表示等待者的數(shù)量
   starvationThresholdNs = 1e6  // 1ms 進(jìn)入饑餓模式的等待時(shí)間閾值
)

而 sema 的含義比較簡單,就是一個(gè)用作不同 goroutine 同步的信號(hào)量。

go 的 Mutex 實(shí)現(xiàn)中,state 字段是一個(gè) 32 位的整數(shù),不同的位記錄了四種不同信息,在這種情況下, 只需要通過原子操作就可以保證一次性實(shí)現(xiàn)對(duì)四種不同狀態(tài)信息的更改,而不需要更多額外的同步機(jī)制。

但是毋庸置疑,這種實(shí)現(xiàn)會(huì)大大降低代碼的可讀性,因?yàn)橥ㄟ^一個(gè)整數(shù)來記錄不同的信息, 就意味著,需要通過各種位運(yùn)算來實(shí)現(xiàn)對(duì)這個(gè)整數(shù)不同位的修改。

當(dāng)然,這只是 Mutex 實(shí)現(xiàn)中最簡單的一種位運(yùn)算了。下面以 state 記錄的四種不同信息為維度來具體講解一下:

mutexLocked:這是 state 的最低位,1 表示鎖被占用,0 表示鎖沒有被占用。

new := mutexLocked 新狀態(tài)為上鎖狀態(tài)

mutexWoken: 這是表示是否有協(xié)程被喚醒了的狀態(tài)

  • new = (old - 1<<mutexWaiterShift) | mutexWoken 等待者數(shù)量減去 1 的同時(shí),設(shè)置喚醒標(biāo)識(shí)
  • new &^= mutexWoken 清除喚醒標(biāo)識(shí)

mutexStarving:饑餓模式的標(biāo)識(shí)

new |= mutexStarving 設(shè)置饑餓標(biāo)識(shí)

等待者數(shù)量:state >> mutexWaiterShift 就是等待者的數(shù)量,也就是上面提到的 FIFO 隊(duì)列中 goroutine 的數(shù)量

  • new += 1 << mutexWaiterShift 等待者數(shù)量加 1
  • delta := int32(mutexLocked - 1<<mutexWaiterShift) 上鎖的同時(shí),將等待者數(shù)量減 1

在上面做了這一系列的位運(yùn)算之后,我們會(huì)得到一個(gè)新的 state 狀態(tài),假設(shè)名為 new,那么我們就可以通過 CAS 操作來將 Mutex 的 state 字段更新:

atomic.CompareAndSwapInt32(&m.state, old, new)

通過上面這個(gè)原子操作,我們就可以一次性地更新 Mutex 的 state 字段,也就是一次性更新了四種狀態(tài)信息。

這種通過一個(gè)整數(shù)記錄不同狀態(tài)的寫法在 sync 包其他的一些地方也有用到,比如 WaitGroup 中的 state 字段。

最后,對(duì)于這種操作,我們需要注意的是,因?yàn)槲覀冊趫?zhí)行 CAS 前后是沒有其他什么鎖或者其他的保護(hù)機(jī)制的, 這也就意味著上面的這個(gè) CAS 操作是有可能會(huì)失敗的,那如果失敗了怎么辦呢?

如果失敗了,也就意味著肯定有另外一個(gè) goroutine 率先執(zhí)行了 CAS 操作并且成功了,將 state 修改為了一個(gè)新的值。 這個(gè)時(shí)候,其實(shí)我們前面做的一系列位運(yùn)算得到的結(jié)果實(shí)際上已經(jīng)不對(duì)了,在這種情況下,我們需要獲取最新的 state,然后再次計(jì)算得到一個(gè)新的 state

所以我們會(huì)在源碼里面看到 CAS 操作是寫在 for 循環(huán)里面的。

state的狀態(tài)及枚舉

state狀態(tài)state狀態(tài)枚舉對(duì)應(yīng)二進(jìn)制對(duì)應(yīng)狀態(tài)
mutexUnLockstate=00000未加鎖
mutexLockedstate=10001加鎖
mutexWokenstate=20010喚醒
mutexStarvingstate=40100饑餓
mutexWaiterShiftstate=30011代表位移

在看下面代碼之前,一定要記住這幾個(gè)狀態(tài)之間的 與運(yùn)算 或運(yùn)算,否則代碼里的與運(yùn)算或運(yùn)算

state:   |32|31|...|3|2|1|
         __________/ | |
               |      | |
               |      | mutex的占用狀態(tài)(1被占用,0可用)
               |      |
               |      mutex的當(dāng)前goroutine是否被喚醒
               |
               當(dāng)前阻塞在mutex上的goroutine數(shù)

互斥鎖的作用

互斥鎖是保證同步的一種工具,主要體現(xiàn)在以下2個(gè)方面:

避免多個(gè)線程在同一時(shí)刻操作同一個(gè)數(shù)據(jù)塊 (sum)

可以協(xié)調(diào)多個(gè)線程,以避免它們在同一時(shí)刻執(zhí)行同一個(gè)代碼塊 (sum++)

什么時(shí)候用

需要保護(hù)一個(gè)數(shù)據(jù)或數(shù)據(jù)塊時(shí)

需要協(xié)調(diào)多個(gè)協(xié)程串行執(zhí)行同一代碼塊,避免并發(fā)問題時(shí)

比如 經(jīng)常遇到A給B轉(zhuǎn)賬100元的例子,這個(gè)時(shí)候就可以用互斥鎖來實(shí)現(xiàn)。

注意的坑

1. 不同 goroutine 可以 Unlock 同一個(gè) Mutex,但是 Unlock 一個(gè)無鎖狀態(tài)的 Mutex 就會(huì)報(bào)錯(cuò)。

2. 因?yàn)?mutex 沒有記錄 goroutine_id,所以要避免在不同的協(xié)程中分別進(jìn)行上鎖/解鎖操作,不然很容易造成死鎖。

建議: 先 Lock 再 Unlock、兩者成對(duì)出現(xiàn)。

3. Mutex 不是可重入鎖

Mutex 不會(huì)記錄持有鎖的協(xié)程的信息,所以如果連續(xù)兩次 Lock 操作,就直接死鎖了。

如何實(shí)現(xiàn)可重入鎖?記錄上鎖的 goroutine 的唯一標(biāo)識(shí),在重入上鎖/解鎖的時(shí)候只需要增減計(jì)數(shù)。

type RecursiveMutex struct {
   sync.Mutex
   owner     int64 // 當(dāng)前持有鎖的 goroutine id // 可以換成其他的唯一標(biāo)識(shí)
   recursion int32 // 這個(gè) goroutine 重入的次數(shù)
}
func (m *RecursiveMutex) Lock() {
   gid := goid.Get()  // 獲取唯一標(biāo)識(shí)
   // 如果當(dāng)前持有鎖的 goroutine 就是這次調(diào)用的 goroutine,說明是重入
   if atomic.LoadInt64(&m.owner) == gid {
      m.recursion++
      return
   }
   m.Mutex.Lock()
   // 獲得鎖的 goroutine 第一次調(diào)用,記錄下它的 goroutine id,調(diào)用次數(shù)加1
   atomic.StoreInt64(&m.owner, gid)
   m.recursion = 1
}
func (m *RecursiveMutex) Unlock() {
   gid := goid.Get()
   // 非持有鎖的 goroutine 嘗試釋放鎖,錯(cuò)誤的使用
   if atomic.LoadInt64(&m.owner) != gid {
      panic(fmt.Sprintf("wrong the owner(%d): %d!", m.owner, gid))
   }
   // 調(diào)用次數(shù)減1
   m.recursion--
   if m.recursion != 0 { // 如果這個(gè) goroutine 還沒有完全釋放,則直接返回
      return
   }
   // 此 goroutine 最后一次調(diào)用,需要釋放鎖
   atomic.StoreInt64(&m.owner, -1)
   m.Mutex.Unlock()
}

4.多高的 QPS 才能讓 Mutex 產(chǎn)生強(qiáng)烈的鎖競爭?

模擬一個(gè) 10ms 的接口,接口邏輯中使用全局共享的 Mutex,會(huì)發(fā)現(xiàn)在較低 QPS 的時(shí)候就開始產(chǎn)生激烈的鎖競爭(打印鎖等待時(shí)間和接口時(shí)間)。

解決方式:首先要盡量避免使用 Mutex。如果要使用 Mutex,盡量多聲明一些 Mutex,采用取模分片的方式去使用其中一個(gè) Mutex 進(jìn)行資源控制。避免一個(gè) Mutex 對(duì)應(yīng)過多的并發(fā)。

簡單總結(jié):壓測或者流量高的時(shí)候發(fā)現(xiàn)系統(tǒng)不正常,打開 pprof 發(fā)現(xiàn) goroutine 指標(biāo)在飆升,并且大量 Goroutine 都阻塞在 Mutex 的 Lock 上,這種現(xiàn)象下基本就可以確定是鎖競爭。

5. Mutex 千萬不能被復(fù)制

因?yàn)閺?fù)制的時(shí)候會(huì)將原鎖的 state 值也進(jìn)行復(fù)制。復(fù)制之后,一個(gè)新 Mutex 可能莫名處于持有鎖、喚醒或者饑餓狀態(tài),甚至等阻塞等待數(shù)量遠(yuǎn)遠(yuǎn)大于0。而原鎖 Unlock 的時(shí)候,卻不會(huì)影響復(fù)制鎖。

關(guān)于鎖的使用建議

寫業(yè)務(wù)時(shí)不能全局使用同一個(gè) Mutex

千萬不要將要加鎖和解鎖分到兩個(gè)以上 Goroutine 中進(jìn)行(容易形成死鎖)

Mutex 千萬不能被復(fù)制(包括不能通過函數(shù)參數(shù)傳遞),否則會(huì)復(fù)制傳參前鎖的狀態(tài):已鎖定 or 未鎖定。很容易產(chǎn)生死鎖,關(guān)鍵是編譯器還發(fā)現(xiàn)不了這個(gè) Deadlock~

盡量避免使用 Mutex,如果非使用不可,盡量多聲明一些 Mutex,采用取模分片的方式去使用其中一個(gè) Mutex(分段鎖)(盡量減小鎖的顆粒度)

參考

標(biāo)準(zhǔn)庫文檔 —— sync.Mutex

以上就是Golang Mutex錯(cuò)過會(huì)后悔的重要知識(shí)點(diǎn)分享的詳細(xì)內(nèi)容,更多關(guān)于Golang Mutex的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • golang規(guī)則引擎gengine用法案例

    golang規(guī)則引擎gengine用法案例

    這篇文章主要為大家介紹了golang?規(guī)則引擎gengine用法案例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • Golang使用sqlite3數(shù)據(jù)庫實(shí)現(xiàn)CURD操作

    Golang使用sqlite3數(shù)據(jù)庫實(shí)現(xiàn)CURD操作

    這篇文章主要為大家詳細(xì)介紹了Golang使用sqlite3數(shù)據(jù)庫實(shí)現(xiàn)CURD操作的相關(guān)知識(shí),文中的示例代碼簡潔易懂,有需要的小伙伴可以參考一下
    2025-03-03
  • 基于Go語言開發(fā)一個(gè)編解碼工具

    基于Go語言開發(fā)一個(gè)編解碼工具

    這篇文章主要為大家詳細(xì)介紹了如何基于Go語言開發(fā)一個(gè)編解碼工具,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以跟隨小編一起了解一下
    2025-03-03
  • 詳解golang開發(fā)中http請求redirect的問題

    詳解golang開發(fā)中http請求redirect的問題

    這篇文章主要介紹了詳解golang開發(fā)中http請求redirect的問題,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • Golang 使用gorm添加數(shù)據(jù)庫排他鎖,for update

    Golang 使用gorm添加數(shù)據(jù)庫排他鎖,for update

    這篇文章主要介紹了Golang 使用gorm添加數(shù)據(jù)庫排他鎖,for update,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Golang 定時(shí)器的終止與重置實(shí)現(xiàn)

    Golang 定時(shí)器的終止與重置實(shí)現(xiàn)

    在實(shí)際開發(fā)過程中,我們有時(shí)候需要編寫一些定時(shí)任務(wù)。很多人都熟悉定時(shí)器的使用,那么定時(shí)器應(yīng)該如何終止與重置,下面我們就一起來了解一下
    2021-08-08
  • Go語言七篇入門教程三函數(shù)方法及接口

    Go語言七篇入門教程三函數(shù)方法及接口

    這篇文章主要為大家介紹了Go語言的函數(shù)方法及接口的示例詳解,本文是Go語言七篇入門系列文章,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2021-11-11
  • vscode如何debug調(diào)試golang代碼

    vscode如何debug調(diào)試golang代碼

    古話說工欲善其事必先利其器,Go語言程序的開發(fā)者而言,當(dāng)下最火的IDE應(yīng)該非微軟的Visual Studio Code莫屬,本文主要介紹了vscode如何debug調(diào)試golang代碼,感興趣的可以了解一下
    2024-03-03
  • 使用Go語言創(chuàng)建靜態(tài)文件服務(wù)器問題

    使用Go語言創(chuàng)建靜態(tài)文件服務(wù)器問題

    這篇文章主要介紹了使用Go語言創(chuàng)建靜態(tài)文件服務(wù)器,本文通過試了代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-03-03
  • Go?easyjson使用及反射原理

    Go?easyjson使用及反射原理

    這篇文章主要介紹了Go?easyjson使用技巧,詳細(xì)介紹了go自帶JSON庫使用的反射原理,性能相對(duì)較差,可以使用easyjson代替,需要的朋友可以參考下
    2022-04-04

最新評(píng)論