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

快速解決Golang Map 并發(fā)讀寫安全的問題

 更新時(shí)間:2020年12月23日 09:46:25   作者:追風(fēng)2019  
這篇文章主要介紹了快速解決Golang Map 并發(fā)讀寫安全的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧

一、錯(cuò)誤案例

package main
import (
	"fmt"
	"time"
)
var TestMap map[string]string
func init() {
	TestMap = make(map[string]string, 1)
}
func main() {
	for i := 0; i < 1000; i++ {
		go Write("aaa")
		go Read("aaa")
		go Write("bbb")
		go Read("bbb")
	}
	time.Sleep(5 * time.Second)
}
func Read(key string) {
	fmt.Println(TestMap[key])
}
func Write(key string) {
	TestMap[key] = key
}

上面代碼執(zhí)行大概率出現(xiàn)報(bào)錯(cuò):fatal error: concurrent map writes

二、問題分析

網(wǎng)上關(guān)于 golang 編程中 map 并發(fā)讀寫相關(guān)的資料很多,但總是都說成 并發(fā)讀寫 造成上面的錯(cuò)誤,到底是 并發(fā)讀 還是 并發(fā)寫 造成的,這個(gè)很多資料都沒有說明。

我們把上面的案例分別在循環(huán)中注釋 Read 和 Write 函數(shù)的調(diào)用,分別測試 并發(fā)讀 和 并發(fā)寫;

循環(huán)次數(shù)分別測試了 100、1 w、100 w 次,并發(fā)讀操作絕對不會報(bào)上面的錯(cuò),而并發(fā)寫基本都會報(bào)錯(cuò)。

因此,這個(gè)錯(cuò)誤主要原因是:map 并發(fā)寫。

三、問題原因

為什么 map 并發(fā)寫會導(dǎo)致這個(gè)錯(cuò)誤? 網(wǎng)絡(luò)上的相關(guān)文章也大都有說明。

因?yàn)?map 變量為 指針類型變量,并發(fā)寫時(shí),多個(gè)協(xié)程同時(shí)操作一個(gè)內(nèi)存,類似于多線程操作同一個(gè)資源會發(fā)生競爭關(guān)系,共享資源會遭到破壞,因此golang 出于安全的考慮,拋出致命錯(cuò)誤:fatal error: concurrent map writes。

四、解決方案

網(wǎng)上各路資料解決方案較多,主要思路是通過加鎖保證每個(gè)協(xié)程同步操作內(nèi)存。

github 上找到一個(gè) concurrentMap 包,案例代碼修改如下:

package main
import (
 "fmt"
 cmap "github.com/orcaman/concurrent-map"
 "time"
)
var TestMap cmap.ConcurrentMap
func init() {
 TestMap = cmap.New()
}
func main() {
 for i := 0; i < 100; i++ {
 go Write("aaa", "111")
 go Read("aaa")
 go Write("bbb", "222")
 go Read("bbb")
 }
 time.Sleep(5 * time.Second)
}
func Read(key string) {
 if v, ok := TestMap.Get(key); ok {
 fmt.Printf("鍵值為 %s 的值為:%s", key, v)
 } else {
 fmt.Printf("鍵值不存在")
 }
}
func Write(key string, value string) {
 TestMap.Set(key, value)
}

五、思考總結(jié)

因?yàn)槲沂且?PHP 打開的編程世界,PHP 語言只有單線程,且不涉及指針操作,變量類型也是弱變量,以 PHP 編程思維剛開始接觸 Golang 時(shí)還比較容易上手,但越往后,語言的特性區(qū)別就體現(xiàn)得越來越明顯,思維轉(zhuǎn)變就越來越大,對我來說是打開了一個(gè)新世界。

像本文出現(xiàn)的錯(cuò)誤案例,也是因?yàn)樽约簺]有多線程編程的思維基礎(chǔ),導(dǎo)致對這種問題不敏感,還是花了蠻多時(shí)間理解的。希望對和我有相似學(xué)習(xí)路線的朋友提供到一些幫助。

補(bǔ)充:Golang Map并發(fā)處理機(jī)制(sync.Map)

Go語言中的Map在并發(fā)情況下,只讀是線程安全的,同時(shí)讀寫線程不安全。

示例:

package main 
import (
 "fmt"
)
var m = make(map[int]int)
func main() {
 //寫入操作
 i:=0
 go func() {
 for{
 i++
 m[1]=i
 }
 
 }()
 //讀操作
 go func() {
 for{
 fmt.Println(m[1])
 }
 
 }()
 //無限循環(huán),讓并發(fā)程序在后臺運(yùn)行
 for {
 ;
 }
}

從以上示例可以看出,不斷地對map進(jìn)行讀和寫,會出現(xiàn)錯(cuò)誤。主要原因是對map進(jìn)行讀和寫發(fā)生了競態(tài)問題。map內(nèi)部會對這種并發(fā)操作進(jìn)行檢查并提前發(fā)現(xiàn)。

如果確實(shí)需要對map進(jìn)行并發(fā)讀寫操作,可以采用加鎖機(jī)制、channel同步機(jī)制,但這樣性能并不高。

Go語言在1.9版本中提供了一種效率較高的并發(fā)安全的sync.Map。

sync.Map結(jié)構(gòu)如下:

The zero Map is empty and ready for use. A Map must not be copied after first use.
type Map struct {
 mu Mutex
 misses int
}
 
// Load returns the value stored in the map for a key, or nil if no
// value is present.
// The ok result indicates whether value was found in the map.
func (m *Map) Load(key interface{}) (value interface{}, ok bool) { 
}
 
// Store sets the value for a key.
func (m *Map) Store(key, value interface{}) {
 
}
// LoadOrStore returns the existing value for the key if present.
// Otherwise, it stores and returns the given value.
// The loaded result is true if the value was loaded, false if stored.
func (m *Map) LoadOrStore(key, value interface{}) (actual interface{}, loaded bool) { 
}
 
// Delete deletes the value for a key.
func (m *Map) Delete(key interface{}) { 
} 
 
// Range calls f sequentially for each key and value present in the map.
// If f returns false, range stops the iteration.
//
// Range does not necessarily correspond to any consistent snapshot of the Map's
// contents: no key will be visited more than once, but if the value for any key
// is stored or deleted concurrently, Range may reflect any mapping for that key
// from any point during the Range call.
//
// Range may be O(N) with the number of elements in the map even if f returns
// false after a constant number of calls.
func (m *Map) Range(f func(key, value interface{}) bool) { 
}
 
func (m *Map) missLocked() {
 
}
 
func (m *Map) dirtyLocked() {
 
}

其實(shí),sync.Map內(nèi)部還是進(jìn)行了加鎖機(jī)制,不過進(jìn)行了一定的優(yōu)化。

sync.Map使用示例:

package main 
import (
 "fmt"
 "sync"
 "time"
)
 
var m1 sync.Map 
func main() {
 i := 0
 go func() {
 for {
 i++
 m1.Store(1, i)
 time.Sleep(1000)
 }
 }()
 go func() {
 for{
 time.Sleep(1000)
 fmt.Println(m1.Load(1))
 }
 
 }()
 for {
 ;
 }
}

成功運(yùn)行效果如下:

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

相關(guān)文章

  • Gin與Mysql實(shí)現(xiàn)簡單Restful風(fēng)格API實(shí)戰(zhàn)示例詳解

    Gin與Mysql實(shí)現(xiàn)簡單Restful風(fēng)格API實(shí)戰(zhàn)示例詳解

    這篇文章主要為大家介紹了Gin與Mysql實(shí)現(xiàn)簡單Restful風(fēng)格API示例詳解,有需要的朋友可以借鑒參考下希望能夠有所幫助,祝大家多多進(jìn)步
    2021-11-11
  • Go使用sync.Pool提高性能的代碼示例

    Go使用sync.Pool提高性能的代碼示例

    在高性能應(yīng)用程序中,頻繁的內(nèi)存分配和回收是性能瓶頸的常見原因之一,Go 語言提供了 sync.Pool 類型,它可以用來存儲和重用臨時(shí)對象,本文將詳細(xì)介紹如何在 Go 中使用 sync.Pool,并通過實(shí)際代碼示例來展示其對性能的提升效果,需要的朋友可以參考下
    2024-04-04
  • win7下配置GO語言環(huán)境 + eclipse配置GO開發(fā)

    win7下配置GO語言環(huán)境 + eclipse配置GO開發(fā)

    這篇文章主要介紹了win7下配置GO語言環(huán)境 + eclipse配置GO開發(fā),需要的朋友可以參考下
    2014-10-10
  • 淺談Go語言的高效編碼細(xì)節(jié)

    淺談Go語言的高效編碼細(xì)節(jié)

    這篇文章主要介紹了淺談Go語言的高效編碼細(xì)節(jié),我們都知道golang是天生的高并發(fā),高效的編譯型語言,可我們也都可知道,工具再好,用法不對,全都白費(fèi),我們來舉2個(gè)常用路徑來感受一下
    2023-01-01
  • 詳解Go中defer與return的執(zhí)行順序

    詳解Go中defer與return的執(zhí)行順序

    Go?defer中改變r(jià)eturn的值會生效嗎,這就設(shè)計(jì)到了GO語言中defer與return哪個(gè)先執(zhí)行的問題了,下面小編就通過簡單的示例來和大家講講吧
    2023-07-07
  • VS Code配置Go語言開發(fā)環(huán)境的詳細(xì)教程

    VS Code配置Go語言開發(fā)環(huán)境的詳細(xì)教程

    這篇文章主要介紹了VS Code配置Go語言開發(fā)環(huán)境的詳細(xì)教程,本文通過實(shí)例代碼圖文相結(jié)合的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-05-05
  • golang中一種不常見的switch語句寫法示例詳解

    golang中一種不常見的switch語句寫法示例詳解

    這篇文章主要介紹了golang中一種不常見的switch語句寫法,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-05-05
  • go語言單例模式(Singleton)實(shí)例分析

    go語言單例模式(Singleton)實(shí)例分析

    這篇文章主要介紹了go語言單例模式(Singleton),實(shí)例分析了單例模式的原理與Go語言的實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2015-03-03
  • goland 實(shí)現(xiàn)自動格式化代碼

    goland 實(shí)現(xiàn)自動格式化代碼

    這篇文章主要介紹了goland 實(shí)現(xiàn)自動格式化代碼的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Golang并發(fā)控制之errgroup使用詳解

    Golang并發(fā)控制之errgroup使用詳解

    errgroup?是?Go?官方庫?x?中提供的一個(gè)非常實(shí)用的工具,用于并發(fā)執(zhí)行多個(gè)?goroutine,并且方便的處理錯(cuò)誤,下面就跟隨小編一起來了解下的它的具體使用吧
    2024-11-11

最新評論