Go語言中map使用和并發(fā)安全詳解
1 map使用
1.1 map定義
map是一種無序的集合,對(duì)應(yīng)的key (索引)會(huì)對(duì)應(yīng)一個(gè)value(值),所以這個(gè)結(jié)構(gòu)也稱為關(guān)聯(lián)數(shù)組或字典。
map在其他語言中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)
}
}
//沒有設(shè)置v值的時(shí)候,map的遍歷是隨機(jī)的,起始遍歷是個(gè)隨機(jī)值
執(zhí)行第一次:

執(zhí)行第二次:

注意:map在增加值、刪除時(shí)需要加互斥鎖
2 并發(fā)安全
Go語言中的 map 在并發(fā)情況下,只讀是線程安全的,同時(shí)讀寫是線程不安全的。
2.1 不安全原因
官網(wǎng)解釋:同一個(gè)變量在多個(gè)goroutine中訪問需要保證其安全性。
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ā)寫時(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)在寫操作的時(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語言中map使用和并發(fā)安全的文章就介紹到這了,更多相關(guān)Go語言map并發(fā)安全內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
goland把go項(xiàng)目打包進(jìn)docker鏡像的全過程記錄
golang編譯的應(yīng)用是不需要依賴其他運(yùn)行環(huán)境的,下面這篇文章主要給大家介紹了關(guān)于goland把go項(xiàng)目打包進(jìn)docker鏡像的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08
Go使用Protocol?Buffers在數(shù)據(jù)序列化的優(yōu)勢(shì)示例詳解
這篇文章主要為大家介紹了Go使用Protocol?Buffers在數(shù)據(jù)序列化的優(yōu)勢(shì)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
Golang優(yōu)雅關(guān)閉channel的方法示例
Goroutine和channel是Go在“并發(fā)”方面兩個(gè)核心feature,下面這篇文章主要給大家介紹了關(guān)于Golang如何優(yōu)雅關(guān)閉channel的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考解決,下面來一起看看吧。2017-11-11
Golang中map的三種聲明定義方式實(shí)現(xiàn)
本文主要介紹了Golang中map的三種聲明定義方式實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02

