濮阳杆衣贸易有限公司

主頁(yè) > 知識(shí)庫(kù) > 淺談golang并發(fā)操作變量安全的問(wèn)題

淺談golang并發(fā)操作變量安全的問(wèn)題

熱門標(biāo)簽:上海極信防封電銷卡價(jià)格 地圖標(biāo)注免費(fèi)定制店 重慶慶云企業(yè)400電話到哪申請(qǐng) 不封卡外呼系統(tǒng) 鄭州智能語(yǔ)音電銷機(jī)器人價(jià)格 湛江crm外呼系統(tǒng)排名 宿遷便宜外呼系統(tǒng)代理商 寧波語(yǔ)音外呼系統(tǒng)公司 仙桃400電話辦理

我就廢話不多說(shuō)了,大家還是直接看代碼吧~

package main 
import (
	"fmt"
	"time"
	"sync"
	"sync/atomic"
)
 
func main() {
	test1()
	test2()
}
 
func test1() {
	var wg sync.WaitGroup
	count := 0
	t := time.Now()
	for i := 0 ; i  50000 ; i++ {
		wg.Add(1)
		go func(wg *sync.WaitGroup,i int) {
			count++ //count不是并發(fā)安全的
			wg.Done()
		}(wg,i)
	}
 
	wg.Wait()
	fmt.Println(time.Now().Sub(t))
	fmt.Println("count====>",count) //count的值50000
	fmt.Println("exit")
} 
 
func test2() {
	var wg sync.WaitGroup
	count := int64(0)
	t := time.Now()
	for i := 0 ; i  50000 ; i++ {
		wg.Add(1)
		go func(wg *sync.WaitGroup,i int) {
			atomic.AddInt64(count,1) //原子操作
			wg.Done()
		}(wg,i)
	}
 
	wg.Wait()
	fmt.Println(time.Now().Sub(t))
	fmt.Println("count====>",count) //count的值為50000
	fmt.Println("exit")
}

執(zhí)行結(jié)果:

18.0485ms
count====> 46621
exit
16.0418ms
count====> 50000
exit

補(bǔ)充:golang 基于共享變量的并發(fā)

并發(fā)定義:當(dāng)我們沒(méi)有辦法自信地確認(rèn)一個(gè)事件是在另一個(gè)事件的前面或者后面發(fā)生的話,就說(shuō)明x和y這兩個(gè)事件是并發(fā)的。

并發(fā)安全:如果其所有可訪問(wèn)的方法和操作都是并發(fā)安全的話,那么類型便是并發(fā)安全的。

競(jìng)爭(zhēng)條件:程序在多個(gè)goroutine交叉執(zhí)行操作時(shí),沒(méi)有給出正確的結(jié)果。

只要有

兩個(gè)goroutine并發(fā)訪問(wèn)

同一變量,且至

少其中的一個(gè)是寫操作的時(shí)候就會(huì)發(fā)生數(shù)據(jù)競(jìng)爭(zhēng)。

數(shù)據(jù)競(jìng)爭(zhēng)會(huì)在兩個(gè)以上的goroutine并發(fā)訪問(wèn)相同的變量且至少其中一個(gè)為寫操作時(shí)發(fā)生。

第一種:不要去寫變量,變量直接提前初始化。

第二種:多個(gè)只允許一個(gè)goroutine訪問(wèn)變量,用select來(lái)監(jiān)聽(tīng)操作(go的金句:不要通過(guò)共享變量來(lái)通信,通過(guò)通信(channel)來(lái)共享變量)。

第三種:允許過(guò)個(gè)goroutine訪問(wèn)變量,但是同一時(shí)間只允許一個(gè)goroutine訪問(wèn)。

現(xiàn)在我們來(lái)講第三種情況具體操作

golang 我們可以通過(guò)channel作為計(jì)量器,可以保證可以有多少個(gè)goroutine可以同時(shí)訪問(wèn)。make(chan struct{},1),通過(guò)寫入讀取用阻塞的方式鎖定住指定的代碼塊的訪問(wèn)。

var (
sema = make(chan struct{}, 1) // a binary semaphore guarding balance
balance int
)
func Deposit(amount int) {
sema - struct{}{} // acquire token
balance = balance + amount
-sema // release token
}
func Balance() int {
sema - struct{}{} // acquire token
b := balance
-sema // release token
return b
}

可以保證同一時(shí)刻只有一個(gè)goroutine來(lái)訪問(wèn)。

然而我們可以用sync包中的Mutex來(lái)實(shí)現(xiàn)上面的功能,那就是:

互斥鎖 sync.Mutex

互斥鎖:保證共享變量不會(huì)被并發(fā)訪問(wèn)。

import "sync"
var (
mu sync.Mutex // guards balance
balance int
)
func Deposit(amount int) {
mu.Lock()
balance = balance + amount
mu.Unlock()
}
func Balance() int {
mu.Lock()
b := balance
mu.Unlock()
return b
}

在Lock和Unlock之間的代碼段中的內(nèi)容goroutine可以隨便讀取或者修改,這個(gè)代碼段叫做臨界區(qū)。

注意:一定要釋放鎖(Unlock),不管任何情況,可以利用defer Mutex.Unlock(),一定要注意go里沒(méi)有重入鎖,如果遇到更小原子的操作,考慮分解成不帶鎖功能的小塊函數(shù)

接下來(lái)我們將另一種鎖:讀寫鎖sync.RWMutex

很多情況我們需要保證讀的性能,而互斥鎖會(huì)短暫的阻止其他的goroutine的運(yùn)行,沒(méi)法達(dá)到很好的多并發(fā)效果(多讀單寫),這時(shí)讀寫鎖就可以很好的解決這個(gè)問(wèn)題。

RLock()和RUnlock()獲取和釋放一個(gè)讀取或者共享鎖。RLock只能在臨界區(qū)共享變量沒(méi)有任何寫入操作時(shí)可用。一般來(lái)說(shuō),我們不應(yīng)該假設(shè)邏輯上的只讀函數(shù)/方法也不會(huì)去更新某一些變量。如果沒(méi)法確定,那么久使用互斥鎖(Mutex)

最后我們來(lái)講下內(nèi)存同步的問(wèn)題

var x, y int
go func() {
x = 1 // A1
fmt.Print("y:", y, " ") // A2
}()
go func() {
y = 1 // B1
fmt.Print("x:", x, " ") // B2
}()

上面的例子:A1、A2、B1、B2 執(zhí)行循序卻是毫無(wú)規(guī)律

在現(xiàn)代計(jì)算機(jī)中可能會(huì)有一堆處理器,每一個(gè)都會(huì)有其本地緩存(local cache)。為了效率,對(duì)內(nèi)存的寫入一般會(huì)在每一個(gè)處理器中緩沖,并在必要時(shí)一起flush到主存。這種情況下這些數(shù)據(jù)可能會(huì)以與當(dāng)初goroutine寫入順序不同的順序被提交到主存。導(dǎo)致程序運(yùn)行串行了,又同時(shí)串行的代碼訪問(wèn)了共享變量,盡管goroutine A中一定需要觀察到x=1執(zhí)行成功之后才會(huì)去讀取y,但它沒(méi)法確保自己觀察得到goroutine B中對(duì)y的寫入,所以A還可能會(huì)打印出y的一個(gè)舊版的值。

有兩種方法解決:

1.變量限定在goroutine中使用,不訪問(wèn)共享變量

2.用互斥條件訪問(wèn)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • Golang實(shí)現(xiàn)web文件共享服務(wù)的示例代碼
  • 解決golang 關(guān)于全局變量的坑
  • Golang 變量申明的三種方式
  • Golang常用環(huán)境變量說(shuō)明與設(shè)置詳解
  • Golang中的變量學(xué)習(xí)小結(jié)
  • 詳解Golang編程中的常量與變量
  • Golang開(kāi)發(fā)中如何解決共享變量問(wèn)題

標(biāo)簽:安康 青海 物業(yè)服務(wù) 遼寧 電子產(chǎn)品 西雙版納 海南 儋州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《淺談golang并發(fā)操作變量安全的問(wèn)題》,本文關(guān)鍵詞  淺談,golang,并發(fā),操作,變量,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《淺談golang并發(fā)操作變量安全的問(wèn)題》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于淺談golang并發(fā)操作變量安全的問(wèn)題的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    普格县| 巫溪县| 盐源县| 福泉市| 突泉县| 香河县| 永济市| 庆城县| 嘉黎县| 渭南市| 栾川县| 江都市| 酉阳| 汝南县| 灵璧县| 长白| 长沙县| 若羌县| 宁安市| 蒙城县| 天祝| 延寿县| 阳春市| 墨玉县| 昌黎县| 伊宁市| 石阡县| 扎赉特旗| 柘荣县| 北安市| 神池县| 唐海县| 西乌珠穆沁旗| 连江县| 丹东市| 永昌县| 南木林县| 松原市| 天台县| 浦江县| 城口县|