這篇文章給大家分享的是有關(guān)Go 并發(fā)編程之 Mutex是什么的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考。一起跟隨小編過來看看吧。
成都創(chuàng)新互聯(lián)憑借在網(wǎng)站建設(shè)、網(wǎng)站推廣領(lǐng)域領(lǐng)先的技術(shù)能力和多年的行業(yè)經(jīng)驗(yàn),為客戶提供超值的營銷型網(wǎng)站建設(shè)服務(wù),我們始終認(rèn)為:好的營銷型網(wǎng)站就是好的業(yè)務(wù)員。我們已成功為企業(yè)單位、個(gè)人等客戶提供了成都網(wǎng)站制作、成都網(wǎng)站設(shè)計(jì)服務(wù),以良好的商業(yè)信譽(yù),完善的服務(wù)及深厚的技術(shù)力量處于同行領(lǐng)先地位。我們比較常見的大型項(xiàng)目的設(shè)計(jì)中都會(huì)出現(xiàn)并發(fā)訪問問題,并發(fā)就是為了解決數(shù)據(jù)的準(zhǔn)確性,保證同一個(gè)臨界區(qū)的數(shù)據(jù)只能被一個(gè)線程進(jìn)行操作,日常中使用到的并發(fā)場景也是很多的:
上面都是并發(fā)帶來的數(shù)據(jù)準(zhǔn)確性的問題,決絕方案就是使用互斥鎖,也就是今天并發(fā)編程中的所要描述的 Mutex 并發(fā)原語。
互斥鎖 Mutex 就是為了避免并發(fā)競爭建立的并發(fā)控制機(jī)制,其中有個(gè)“臨界區(qū)”的概念。
在并發(fā)編程過程中,如果程序中一部分資源或者變量會(huì)被并發(fā)訪問或者修改,為了避免并發(fā)訪問導(dǎo)致數(shù)據(jù)的不準(zhǔn)確,這部分程序需要率先被保護(hù)起來,之后操作,操作結(jié)束后去除保護(hù),這部分被保護(hù)的程序就叫做臨界區(qū)。
使用互斥鎖,限定臨界區(qū)只能同時(shí)由一個(gè)線程持有,若是臨界區(qū)此時(shí)被一個(gè)線程持有,那么其他線程想進(jìn)入到這個(gè)臨界區(qū)的時(shí)候,就會(huì)失敗或者等待釋放鎖,持有此臨界區(qū)的線程退出,其他線程才有機(jī)會(huì)獲得這個(gè)臨界區(qū)。
go mutex 臨界區(qū)示意圖
Mutex 是 Go 語言中使用最廣泛的同步原語,也稱為并發(fā)原語,解決的是并發(fā)讀寫共享資源,避免出現(xiàn)數(shù)據(jù)競爭 data race 問題。
互斥鎖 Mutex 提供了兩個(gè)方法 Lock 和 Unlock:進(jìn)入到臨界區(qū)使用 Lock 方法加鎖,退出臨界區(qū)使用 Unlock 方法釋放鎖。
type Locker interface { Lock() Unlock()}func(m *Mutex)Lock()func(m *Mutex)Unlock()
當(dāng)一個(gè) goroutine 調(diào)用 Lock 方法獲取到鎖后,其他 goroutine 會(huì)阻塞在 Lock 的調(diào)用上,直到當(dāng)前獲取到鎖的 goroutine 釋放鎖。
接下來是一個(gè)計(jì)數(shù)器的例子,是由 100 個(gè) goroutine 對(duì)計(jì)數(shù)器進(jìn)行累加操作,最后輸出結(jié)果:
package mainimport ( "fmt" "sync")func main() { var mu sync.Mutex countNum := 0 // 確認(rèn)輔助變量是否都執(zhí)行完成 var wg sync.WaitGroup // wg 添加數(shù)目要和 創(chuàng)建的協(xié)程數(shù)量保持一致 wg.Add(100) for i := 0; i < 100; i++ { go func() { defer wg.Done() for j := 0; j < 1000; j++ { mu.Lock() countNum++ mu.Unlock() } }() } wg.Wait() fmt.Printf("countNum: %d", countNum)}
很多時(shí)候 Mutex 并不是單獨(dú)使用的,而是嵌套在 Struct 中使用,作為結(jié)構(gòu)體的一部分,如果嵌入的 struct 有多個(gè)字段,我們一般會(huì)把 Mutex 放在要控制的字段上面,然后使用空格把字段分隔開來。
甚至可以把獲取鎖、釋放鎖、計(jì)數(shù)加一的邏輯封裝成一個(gè)方法。
package mainimport ( "fmt" "sync")// 線程安全的計(jì)數(shù)器type Counter struct { CounterType int Name string mu sync.Mutex count uint64}// 加一方法func (c *Counter) Incr() { c.mu.Lock() defer c.mu.Unlock() c.count++}// 取數(shù)值方法 線程也需要受保護(hù)func (c *Counter) Count() uint64 { c.mu.Lock() defer c.mu.Unlock() return c.count}func main() { // 定義一個(gè)計(jì)數(shù)器 var counter Counter var wg sync.WaitGroup wg.Add(100) for i := 0; i < 100; i++ { go func() { defer wg.Done() for j := 0; j < 1000; j++ { counter.Incr() } }() } wg.Wait() fmt.Printf("%d\n", counter.Count())}
感謝各位的閱讀!關(guān)于Go 并發(fā)編程之 Mutex是什么就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!
名稱欄目:Go并發(fā)編程之Mutex是什么-創(chuàng)新互聯(lián)
本文鏈接:http://vcdvsql.cn/article16/djhhgg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供外貿(mào)建站、網(wǎng)站建設(shè)、網(wǎng)站內(nèi)鏈、網(wǎng)頁設(shè)計(jì)公司、定制網(wǎng)站、Google
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容