亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go并發(fā)與鎖的兩種方式該如何提效詳解

 更新時(shí)間:2022年12月27日 08:55:11   作者:機(jī)智的程序員小熊  
如果沒(méi)有鎖,在我們的項(xiàng)目中,可能會(huì)存在多個(gè)goroutine同時(shí)操作一個(gè)資源(臨界區(qū)),這種情況會(huì)發(fā)生競(jìng)態(tài)問(wèn)題(數(shù)據(jù)競(jìng)態(tài)),下面這篇文章主要給大家介紹了關(guān)于Go并發(fā)與鎖的兩種方式該如何提效的相關(guān)資料,需要的朋友可以參考下

并發(fā)安全,就是多個(gè)并發(fā)體在同一段時(shí)間內(nèi)訪問(wèn)同一個(gè)共享數(shù)據(jù),共享數(shù)據(jù)能被正確處理。

很多語(yǔ)言的并發(fā)編程很容易在同時(shí)修改某個(gè)變量的時(shí)候,因?yàn)椴僮鞑皇窃拥?,而出現(xiàn)錯(cuò)誤計(jì)算,比如一個(gè)加法運(yùn)算使用中的變量被修改,而導(dǎo)致計(jì)算結(jié)果出錯(cuò),典型的像統(tǒng)計(jì)商品庫(kù)存。

個(gè)人建議只要涉及到共享變量統(tǒng)統(tǒng)使用channel,因?yàn)?code>channel源碼中使用了互斥鎖,它是并發(fā)安全的。

我們可以不用,但不可以不了解,手中有糧心中不慌。

并發(fā)不安全的例子

數(shù)組是并發(fā)不安全的,在例子開(kāi)始前我們要知道append函數(shù)的行為:長(zhǎng)度足夠在原數(shù)組cap內(nèi)追加函數(shù),增加len,長(zhǎng)度不夠則觸發(fā)擴(kuò)容,申請(qǐng)新數(shù)組cap增加一倍,賦值遷移。

所以在這個(gè)過(guò)程中,僅討論擴(kuò)容操作的話可能存在同時(shí)申請(qǐng)并賦值的情況,導(dǎo)致漏掉某次擴(kuò)容增加的數(shù)據(jù)。

var s []int

func appendValue(i int) {
	s = append(s, i)
}

func main() {
	for i := 0; i < 10000; i++ { //10000個(gè)協(xié)程同時(shí)添加切片
		go appendValue(i)
	}
    time.Sleep(2)
    fmt.Println(len(s))
}

比如上例,10000 個(gè)協(xié)程同時(shí)為切片增加數(shù)據(jù),你可以嘗試運(yùn)行一下,打印出來(lái)的一定不是 10000

  • 以上并發(fā)操作的同一個(gè)資源,專(zhuān)業(yè)名詞叫做臨界區(qū)。
  • 因?yàn)椴l(fā)操作存在數(shù)據(jù)競(jìng)爭(zhēng),導(dǎo)致數(shù)據(jù)值意外改寫(xiě),最后的結(jié)果與期待的不符,這種問(wèn)題統(tǒng)稱(chēng)為競(jìng)態(tài)問(wèn)題。

常見(jiàn)于控制商品減庫(kù)存,控制余額增減等情況。 那么有什么辦法解決競(jìng)態(tài)問(wèn)題呢?

  • 互斥鎖:讓訪問(wèn)某個(gè)臨界區(qū)的時(shí)候,只有一個(gè) goroutine 可以訪問(wèn)。
  • 原子操作:讓某些操作變成原子的,這個(gè)后續(xù)討論。

這兩個(gè)思路貫穿了無(wú)數(shù)的高并發(fā)/分布式方案,區(qū)別是在一個(gè)進(jìn)程應(yīng)用中使用,還是借助某些第三方手段來(lái)實(shí)現(xiàn),比如中間件。獨(dú)孤九劍森羅萬(wàn)象一定要牢牢記住。

互斥鎖

Go 語(yǔ)言中互斥鎖的用法如下:

var lock sync.Mutex //互斥鎖
lock.Lock() //加鎖
s = append(s, i)
lock.Unlock() //解鎖

在訪問(wèn)臨界區(qū)的前后加上互斥鎖,就可以保證不會(huì)出現(xiàn)并發(fā)問(wèn)題。

我們修改還是上一個(gè)4.7.1的例子,為其增加互斥鎖。

var s []int
var lock sync.Mutex

appendValueSafe := func(i int) {
    lock.Lock()
    s = append(s, i)
    lock.Unlock()
}

for i := 0; i < 10000; i++ { //10000個(gè)協(xié)程同時(shí)添加切片
    go appendValueSafe(i)
}
time.Sleep(2)
fmt.Println(len(s))
  • 對(duì)共享變量s的寫(xiě)入操作加互斥鎖,保證同一時(shí)刻只有一個(gè)goroutine修改內(nèi)容。
  • 加鎖之后到解鎖之前的內(nèi)容,同一時(shí)刻只有一個(gè)訪問(wèn),無(wú)論讀寫(xiě)。
  • 無(wú)論多少次都輸出10000,不會(huì)再出現(xiàn)競(jìng)態(tài)問(wèn)題。
  • 要注意:如果在寫(xiě)的同時(shí),有并發(fā)讀操作時(shí),為了防止不要讀取到寫(xiě)了一半數(shù)據(jù),需要為讀操作也加鎖。

讀寫(xiě)鎖

互斥鎖是完全互斥的,并發(fā)讀沒(méi)有修改的情況下是不會(huì)有問(wèn)題的,也沒(méi)有必要在并發(fā)讀的時(shí)候加鎖不然效率會(huì)變低。

用法:

rwlock sync.RWMutex
//讀鎖
rwlock.RLock()
rwlock.RUnlock()

//寫(xiě)鎖
rwlock.Lock()
rwlock.Unlock()

并發(fā)讀不互斥可以同時(shí),在一個(gè)寫(xiě)鎖獲取時(shí),其他所有鎖都等待, 口訣:讀讀不互斥、讀寫(xiě)互斥、寫(xiě)寫(xiě)互斥。具體測(cè)算速度的代碼可以見(jiàn)4.7.3的源碼,感興趣的可以改下注釋位置看下效率是有很明顯的提升的。

小結(jié)

  • 學(xué)習(xí)了幾個(gè)名詞:臨界區(qū)、競(jìng)態(tài)問(wèn)題、互斥鎖、原子操作、讀寫(xiě)鎖。
  • 互斥鎖:sync.Mutex, 讀寫(xiě)鎖:sync.RWMutex 都是 sync 包的。
  • 讀寫(xiě)鎖比互斥鎖效率高。

問(wèn)題:只加寫(xiě)鎖可以嗎?為什么?

總結(jié)

到此這篇關(guān)于Go并發(fā)與鎖的兩種方式該如何提效的文章就介紹到這了,更多相關(guān)Go并發(fā)與鎖提效內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go strconv包實(shí)現(xiàn)字符串和基本數(shù)據(jù)類(lèi)型轉(zhuǎn)換的實(shí)例詳解

    Go strconv包實(shí)現(xiàn)字符串和基本數(shù)據(jù)類(lèi)型轉(zhuǎn)換的實(shí)例詳解

    在Go語(yǔ)言(Golang)的編程實(shí)踐中,strconv包是一個(gè)非常重要的標(biāo)準(zhǔn)庫(kù),它提供了在基本數(shù)據(jù)類(lèi)型(如整型、浮點(diǎn)型、布爾型)和字符串之間的轉(zhuǎn)換功能,本文給大家介紹了關(guān)于Go語(yǔ)言字符串轉(zhuǎn)換strconv,需要的朋友可以參考下
    2024-09-09
  • golang json性能分析詳解

    golang json性能分析詳解

    json格式可以算我們?nèi)粘W畛S玫男蛄谢袷街涣耍珿o語(yǔ)言作為一個(gè)由Google開(kāi)發(fā),號(hào)稱(chēng)互聯(lián)網(wǎng)的C語(yǔ)言的語(yǔ)言,自然也對(duì)JSON格式支持很好。下面這篇文章主要給大家詳細(xì)分析介紹了golang json性能的相關(guān)資料,需要的朋友可以參考下。
    2018-02-02
  • golang 切片截取參數(shù)方法詳解

    golang 切片截取參數(shù)方法詳解

    這篇文章主要介紹了golang 切片截取參數(shù)方法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • Golang編程并發(fā)工具庫(kù)MapReduce使用實(shí)踐

    Golang編程并發(fā)工具庫(kù)MapReduce使用實(shí)踐

    這篇文章主要為大家介紹了Golang并發(fā)工具庫(kù)MapReduce的使用實(shí)踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-04-04
  • GO語(yǔ)言延遲函數(shù)defer用法詳解

    GO語(yǔ)言延遲函數(shù)defer用法詳解

    本文主要介紹了GO語(yǔ)言延遲函數(shù)defer用法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • 關(guān)于golang test緩存問(wèn)題

    關(guān)于golang test緩存問(wèn)題

    這篇文章主要介紹了關(guān)于golang test緩存問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-09-09
  • Go語(yǔ)言中工作池的原理與實(shí)現(xiàn)

    Go語(yǔ)言中工作池的原理與實(shí)現(xiàn)

    工作池是一種并發(fā)編程模式,它使用一組固定數(shù)量的工作線程來(lái)執(zhí)行任務(wù)隊(duì)列中的工作單元,本文將介紹工作池的工作原理,并通過(guò)代碼示例演示其在實(shí)際應(yīng)用中的用途,有需要的可以參考下
    2023-10-10
  • 基于context.Context的Golang?loader緩存請(qǐng)求放大問(wèn)題解決

    基于context.Context的Golang?loader緩存請(qǐng)求放大問(wèn)題解決

    這篇文章主要為大家介紹了基于context.Context的Golang?loader緩存請(qǐng)求放大解決方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • golang利用函數(shù)閉包實(shí)現(xiàn)簡(jiǎn)單的中間件

    golang利用函數(shù)閉包實(shí)現(xiàn)簡(jiǎn)單的中間件

    中間件設(shè)計(jì)模式是一種常見(jiàn)的軟件設(shè)計(jì)模式,它在許多編程語(yǔ)言和框架中被廣泛應(yīng)用,這篇文章主要為大家介紹一下golang利用函數(shù)閉包實(shí)現(xiàn)一個(gè)簡(jiǎn)單的中間件,感興趣的可以了解下
    2023-10-10
  • golang rate令牌桶源碼分析實(shí)現(xiàn)方式

    golang rate令牌桶源碼分析實(shí)現(xiàn)方式

    這篇文章主要介紹了golang rate令牌桶源碼分析實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07

最新評(píng)論