Go語(yǔ)言中map使用和并發(fā)安全詳解
1 map使用
1.1 map定義
map是一種無(wú)序的集合,對(duì)應(yīng)的key (索引)會(huì)對(duì)應(yīng)一個(gè)value(值),所以這個(gè)結(jié)構(gòu)也稱為關(guān)聯(lián)數(shù)組或字典。
map在其他語(yǔ)言中hash、hash table等
var mapname map[keytype]valuetype
- mapname 為 map 的變量名。
- keytype 為鍵類型。
- valuetype 是鍵對(duì)應(yīng)的值類型。
1.2 map的使用和概念
map是引用類型,未初始化的map是nil
package main
import "fmt"
func main() {
var maplist map[string]int
maplist["one"] = 1
fmt.Println(maplist)
}
//報(bào)錯(cuò):panic: assignment to entry in nil map
//map需要先初始化內(nèi)存后使用

正確做法:
package main
import "fmt"
func main() {
var maplist map[string]int
maplist = map[string]int{"one": 1, "two": 2}
maplist["three"] = 3
fmt.Println(maplist)
}
//map[one:1 three:3 two:2]

當(dāng)然也可以這樣子:
package main
import "fmt"
func main() {
maplist := make(map[string]int)//初始化內(nèi)存了,想賦值就賦值
maplist["three"] = 3
fmt.Println(maplist)
}

map必須先初始化內(nèi)存,后使用,也就是需要make一下,或者直接賦值一個(gè)空map
maplist := map[string]int{}
fmt.Println(maplist)
1.3 map的容量
和數(shù)組不同的是,map可以根據(jù)新增的key-value動(dòng)態(tài)的伸縮,因此不存在固定長(zhǎng)度或者最大限制,但是也可以選擇初始化容量的值
maplist := make(map[string]float, 100)
出于性能考慮,對(duì)于大的map或者快速擴(kuò)張的map,最好先標(biāo)明
用切片作為map的值
maplist1 := make(map[int][]int) maplist2 := make(map[int]*[]int)
golang里的類型使用靈活,也可以任意組合,map里的值可以是struct,也可以是int、string、甚至是切片、數(shù)組。
1.4 map的使用
1.4.1 map的遍歷
scene := make(map[string]int)
scene["route"] = 66
scene["brazil"] = 4
scene["china"] = 960
for k, v := range scene {
fmt.Println(k, v)
}
1.4.2 map的刪除和斷言
package main
import "fmt"
func main() {
maplist := make(map[string]int)
// 準(zhǔn)備map數(shù)據(jù)
maplist["LYY"] = 66
maplist["520"] = 4
maplist["666"] = 960
delete(maplist, "666")
for k, v := range maplist {
fmt.Println(k, v)
}
}

1.5 map的坑
package main
import "fmt"
func main() {
m := map[int]struct{}{
1: {},
2: {},
3: {},
4: {},
5: {},
}
for k := range m {
fmt.Println(k)
}
}
//沒(méi)有設(shè)置v值的時(shí)候,map的遍歷是隨機(jī)的,起始遍歷是個(gè)隨機(jī)值
執(zhí)行第一次:

執(zhí)行第二次:

注意:map在增加值、刪除時(shí)需要加互斥鎖
2 并發(fā)安全
Go語(yǔ)言中的 map 在并發(fā)情況下,只讀是線程安全的,同時(shí)讀寫(xiě)是線程不安全的。
2.1 不安全原因
官網(wǎng)解釋:同一個(gè)變量在多個(gè)goroutine中訪問(wèn)需要保證其安全性。
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
}
//報(bào)錯(cuò) fatal error: concurrent map writes
原因:因?yàn)閙ap變量為 指針類型變量,并發(fā)寫(xiě)時(shí),多個(gè)協(xié)程同時(shí)操作一個(gè)內(nèi)存,類似于多線程操作同一個(gè)資源會(huì)發(fā)生競(jìng)爭(zhēng)關(guān)系,共享資源會(huì)遭到破壞,因此golang 出于安全的考慮,拋出致命錯(cuò)誤:fatal error: concurrent map writes。
2.2 解決方案
(1)在寫(xiě)操作的時(shí)候增加鎖,刪除時(shí)候除了加鎖外,還需要增加斷言避免出現(xiàn)錯(cuò)誤
package main
import (
"fmt"
"sync"
)
func main() {
var lock sync.Mutex
var maplist map[string]int
maplist = map[string]int{"one": 1, "two": 2}
lock.Lock()
maplist["three"] = 3
lock.Unlock()
fmt.Println(maplist)
}
執(zhí)行結(jié)果:

(2)sync.Map包
package main
import (
"fmt"
"sync"
)
func main() {
m := sync.Map{} //或者 var mm sync.Map
m.Store("a", 1)
m.Store("b", 2)
m.Store("c", 3) //插入數(shù)據(jù)
fmt.Println(m.Load("a")) //讀取數(shù)據(jù)
m.Range(func(key, value interface{}) bool { //遍歷
fmt.Println(key, value)
return true
})
}
執(zhí)行結(jié)果:

我們稱其為并發(fā)安全的map。
總結(jié)
到此這篇關(guān)于Go語(yǔ)言中map使用和并發(fā)安全的文章就介紹到這了,更多相關(guān)Go語(yǔ)言map并發(fā)安全內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go Slice進(jìn)行參數(shù)傳遞如何實(shí)現(xiàn)詳解
這篇文章主要為大家介紹了Go Slice進(jìn)行參數(shù)傳遞如何實(shí)現(xiàn)的過(guò)程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12
對(duì)Golang中的runtime.Caller使用說(shuō)明
這篇文章主要介紹了對(duì)Golang中的runtime.Caller使用說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12
golang限流庫(kù)兩個(gè)大bug(半年之久無(wú)人提起)
最近我的同事在使用uber-go/ratelimit[1]這個(gè)限流庫(kù)的時(shí)候,遇到了兩個(gè)大?bug,這兩個(gè)?bug?都是在這個(gè)庫(kù)的最新版本(v0.3.0)中存在的,而這個(gè)版本從?7?月初發(fā)布都已經(jīng)過(guò)半年了,都沒(méi)人提?bug,難道大家都沒(méi)遇到過(guò)么2023-12-12
Go語(yǔ)言常見(jiàn)錯(cuò)誤接口污染解決分析
這篇文章主要為大家介紹了Go語(yǔ)言常見(jiàn)錯(cuò)誤接口污染解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
Go習(xí)慣用法(多值賦值短變量聲明賦值簡(jiǎn)寫(xiě)模式)基礎(chǔ)實(shí)例
本文為大家介紹了Go習(xí)慣用法(多值賦值,短變量聲明和賦值,簡(jiǎn)寫(xiě)模式、多值返回函數(shù)、comma,ok 表達(dá)式、傳值規(guī)則)的基礎(chǔ)實(shí)例,幫大家鞏固扎實(shí)Go語(yǔ)言基礎(chǔ)2024-01-01
Golang Http 驗(yàn)證碼示例實(shí)現(xiàn)
這篇文章主要介紹了Golang Http 驗(yàn)證碼示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Goland使用Go Modules創(chuàng)建/管理項(xiàng)目的操作
這篇文章主要介紹了Goland使用Go Modules創(chuàng)建/管理項(xiàng)目的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05

