成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一篇學會 Go 的 TryLock 實現

開發 前端
在 Go 1.18 中,為 sync.Mutex 新增了一個新的方法 TryLock(),它是一種非阻塞模式的取鎖操作。當調用 TryLock() 時,該函數僅簡單地返回 true 或者 false,代表是否加鎖成功。

在并發編程中,為了避免多線程同時讀寫共享資源,我們需要互斥。Go 標準庫提供了互斥鎖 sync.Mutex ,通過加鎖 Lock() 方法和解鎖 Unlock() 方法達到對共享資源的并發控制。

在之前的設計中,當鎖被占有,其他 goroutine 嘗試獲取鎖時會被阻塞。這種方式當然是合理的,但是在某些情況下,或許我們希望在獲取鎖失敗時,并不想停止執行,而是可以進入其他的邏輯。

在 Go 1.18 中,為 sync.Mutex 新增了一個新的方法 TryLock(),它是一種非阻塞模式的取鎖操作。當調用 TryLock() 時,該函數僅簡單地返回 true 或者 false,代表是否加鎖成功。

有了 TryLock 的存在,我們就可以由這樣的代碼:

 m.Lock()
// 阻塞等待加鎖成功后的邏輯

轉變成這樣的邏輯

 if m.TryLock(){
// 加鎖成功的邏輯
}else {
// 加鎖失敗的邏輯
}

TryLock 實現

在Go精妙的互斥鎖設計一文中,我們詳細分析過互斥鎖的設計,其代碼輕量簡潔,通過巧妙的位運算,僅僅采用 state 一個字段就實現了四個字段的效果,非常之精彩,建議感興趣的讀者一讀。

而 TryLock() 的實現更加簡單。

func (m *Mutex) TryLock() bool {
old := m.state
if old&(mutexLocked|mutexStarving) != 0 {
return false
}

// There may be a goroutine waiting for the mutex, but we are
// running now and can try to grab the mutex before that
// goroutine wakes up.
if !atomic.CompareAndSwapInt32(&m.state, old, old|mutexLocked) {
return false
}

if race.Enabled {
race.Acquire(unsafe.Pointer(m))
}
return true
}

當鎖被其他 goroutine 占有,或者當前鎖正處于饑餓模式,它將立即返回 false。

func (m *Mutex) Lock() {
// Fast path: grab unlocked mutex.
if atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked) {
if race.Enabled {
race.Acquire(unsafe.Pointer(m))
}
return
}
// Slow path (outlined so that the fast path can be inlined)
m.lockSlow()
}

而當鎖可用時,TryLock() 會采用與 Lock() 方法一樣的方式去嘗試獲取鎖。但在獲取失敗時,與 Lock() 將不一樣,它不會自旋或者阻塞。這是一個完全的非阻塞獲取方式。

應用場景

正如 TryLock() 方法的注釋一樣,它的應用場景并不常見,并且也不被鼓勵使用。

// Note that while correct uses of TryLock do exist, they are rare,
// and use of TryLock is often a sign of a deeper problem
// in a particular use of mutexes.

在當前 Go1.18 標準庫源碼中,與 Lock() 方法被大量內部使用而截然不同的是,并沒有找到一處使用 TryLock() 的地方,僅僅在測試文件 mutex_test.go 中,有找到該方法的新增測試用例。

這里貼一個 TryLock 的使用場景討論:https://stackoverflow.com/questions/41788074/use-case-for-lock-trylock

另外,在開源社區已經有不少 Go 的 TryLock 實現庫。它們基于 sync.Mutex 通過 CAS 操作和 unsafe 指針實現 ;或者利用 channel 實現。

但是這些庫都不能競態檢測。因此,官方支持實現 TryLock 是必要的,避免 TryLock 被濫用。且由于可以集成競態檢測,相較于三方庫實現,有利于開發者發現問題。

總結

從 2012 年開始,實際上很早就有關于 Go 增加 TryLock 的 issue 討論,但是直到 Go 1.18 才被增加。這其中很大一部分原因是,并沒有合理的案例值得添加 TryLock。

Go Team 的負責人 rsc 之前提出的反對意見:TryLock 會鼓勵開發者對鎖進行不精確的思考,并最終導致競態問題。

另外,Go 1.18 除了為互斥鎖 sync.Mutex 新增了 TryLoc() 方法外,也為讀寫鎖 sync.RWMutex 新增了相應的 TryRLock() 和 TryLock() 方法。

正如新增的這三個方法的注釋,雖然使用它們的情況存在,但很少見,使用需謹慎。

責任編輯:武曉燕 來源: Golang技術分享
相關推薦

2021-07-16 22:43:10

Go并發Golang

2021-12-04 22:05:02

Linux

2022-06-09 08:41:17

Go網絡庫Gnet

2024-05-10 08:15:32

go語言反射機制

2024-04-02 12:36:01

2022-03-17 19:29:04

CSS切角自適應

2023-12-05 07:14:27

AIGo

2021-11-15 10:29:39

Go語言類型

2022-01-02 08:43:46

Python

2022-06-30 22:53:18

數據結構算法

2021-08-01 07:19:16

語言OpenrestyNginx

2022-02-07 11:01:23

ZooKeeper

2021-05-11 08:54:59

建造者模式設計

2021-07-02 09:45:29

MySQL InnoDB數據

2021-07-06 08:59:18

抽象工廠模式

2023-01-03 08:31:54

Spring讀取器配置

2022-08-26 09:29:01

Kubernetes策略Master

2021-07-05 22:11:38

MySQL體系架構

2023-11-28 08:29:31

Rust內存布局

2022-08-23 08:00:59

磁盤性能網絡
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩三级免费网站 | 亚洲精品在线播放 | 欧美一区二区三区大片 | 精品视频在线观看 | 不卡一区二区三区四区 | 午夜看看| 精品伦精品一区二区三区视频 | 国产 日韩 欧美 中文 在线播放 | 荷兰欧美一级毛片 | 伊人网一区 | 久久高清 | 欧美久久免费观看 | 欧美三区在线观看 | 久热伊人 | 欧美成人a | 天天艹日日干 | 日韩在线精品 | 国产精品一区二区三区四区 | 在线观看av免费 | 91佛爷在线观看 | 色吊丝2288sds中文字幕 | 久久com| 深夜福利影院 | 久久亚洲精品久久国产一区二区 | 国产欧美日韩一区二区三区 | a黄视频| 在线视频成人 | 欧美视频免费在线 | 日韩理论电影在线观看 | 日韩一级二级片 | 1区2区3区视频 | 一级毛片成人免费看a | 中文字幕一区二区三区精彩视频 | 成人天堂| 国产在线一级片 | 亚洲日本欧美日韩高观看 | 999久久久久久久久6666 | 视频一区二区三区四区五区 | 亚洲一区二区中文字幕 | 久久久久国产精品 | 天堂精品 |