脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - Go语言中map使用和并发安全详解

Go语言中map使用和并发安全详解

2022-07-20 19:05行走的皮卡丘 Golang

golang 自带的map不是并发安全的,并发读写会报错,所以下面这篇文章主要给大家介绍了关于Go语言中map使用和并发安全的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下

 

1 map使用

 

1.1 map定义

map是一种无序的集合,对应的key (索引)会对应一个value(值),所以这个结构也称为关联数组或字典。

map在其他语言中hash、hash table等

var mapname map[keytype]valuetype

  • mapname 为 map 的变量名。
  • keytype 为键类型。
  • valuetype 是键对应的值类型。

 

1.2 map的使用和概念

map是引用类型,未初始化的map是nil

  1. package main
  2.  
  3. import "fmt"
  4.  
  5. func main() {
  6.      var maplist map[string]int
  7.      maplist["one"] = 1
  8.      fmt.Println(maplist)
  9. }
  10. //报错:panic: assignment to entry in nil map
  11. //map需要先初始化内存后使用

Go语言中map使用和并发安全详解

正确做法

  1. package main
  2.  
  3. import "fmt"
  4.  
  5. func main() {
  6.      var maplist map[string]int
  7.      maplist = map[string]int{"one": 1, "two": 2}
  8.      maplist["three"] = 3
  9.      fmt.Println(maplist)
  10. }
  11. //map[one:1 three:3 two:2]

Go语言中map使用和并发安全详解

当然也可以这样子

  1. package main
  2.  
  3. import "fmt"
  4.  
  5. func main() {
  6.      maplist := make(map[string]int)//初始化内存了,想赋值就赋值
  7.      maplist["three"] = 3
  8.      fmt.Println(maplist)
  9. }

Go语言中map使用和并发安全详解

map必须先初始化内存,后使用,也就是需要make一下,或者直接赋值一个空map

  1. maplist := map[string]int{}
  2. fmt.Println(maplist)

 

1.3 map的容量

和数组不同的是,map可以根据新增的key-value动态的伸缩,因此不存在固定长度或者最大限制,但是也可以选择初始化容量的值

  1. maplist := make(map[string]float, 100)

出于性能考虑,对于大的map或者快速扩张的map,最好先标明

用切片作为map的值

  1. maplist1 := make(map[int][]int)
  2. maplist2 := make(map[int]*[]int)

golang里的类型使用灵活,也可以任意组合,map里的值可以是struct,也可以是int、string、甚至是切片、数组。

 

1.4 map的使用

 

1.4.1 map的遍历python

  1. scene := make(map[string]int)
  2.  
  3. scene["route"] = 66
  4. scene["pythonbrazil"] = 4
  5. s编程cene["china"] = 960
  6.  
  7. for k, v := range scene {
  8.      fmt.Println(k, v)
  9. }

 

1.4.2 map的删除和断言

  1. package main
  2.  
  3. import "fmt"
  4.  
  5. func main() {
  6.      maplist := make(map[string]int)
  7.  
  8.      // 准备map数据
  9.      maplist["LYY"] = 66
  10.      maplist["520"] = 4
  11.      maplist["666"] = 960
  12.  
  13.      delete(maplist, "666")
  14.  
  15.      for k, v := range maplist {
  16.          fmt.Println(k, v)
  17.      }
  18. }

Go语言中map使用和并发安全详解

 

1.5 map的坑

  1. package main
  2.  
  3. import "fmt"
  4.  
  5. func main() {
  6.      m := map[int]struct{}{
  7.          1: {},
  8.          2: {},
  9.          3: {},
  10.          4: {},
  11.          5: {}http://www.cppcns.com,
  12.      }
  13.  
  14.      for k := range m {
  15.          fmt.Println(k)
  16.      }
  17. }
  18. //没有设置v值的时候,map的遍历是随机的,起始遍历是个随机值

执行第一次

Go语言中map使用和并发安全详解

执行第二次

Go语言中map使用和并发安全详解

注意:map在增加值、删除时需要加互斥锁

 

2 并发安全

Go语言中的 map 在并发情况下,只读是线程安全的,同时读写是线程不安全的。

 

2.1 不安全原因

官网解释:同一个变量在多个goroutine中访问需要保证其安全性

  1. package main
  2. import (
  3.      "fmt"
  4.      "time"
  5. )
  6. var TestMap map[string]string
  7. func init() {
  8.      TestMap = make(map[string]string, 1)
  9. }
  10. func main() {
  11.      for i := 0; i < 1000; i++ {
  12.          go Write("aaa")
  13.          go Read("aaa")
  14.          go Write("bbb")
  15.          go Read("bbb")
  16.      }
  17.      time.Sleep(5 * time.Second)
  18. }
  19. func Read(key string) {
  20.      fmt.Println(TestMap[key])
  21. }
  22. func Write(key string) {
  23.      TestMap[key] = key
  24. }
  25. //报错 fatal error: c编程oncurrent map writes
  26.  

原因:因为map变量为 指针类型变量,并发写时,多个协程同时操作一个内存,类似于多线程操作同一个资源会发生竞争关系,共享资源会遭到破坏,因此golang 出于安全的考虑,抛出致命错误:fatal error: concurrent map writes。

 

2.2 解决方案

(1)在写操作的时候增加锁,删除时候除了加锁外,还需要增加断言避免出现错误

  1. package main
  2.  
  3. import (
  4.      "fmt"
  5.      "sync"
  6. )
  7.  
  8. func main() {
  9.      var lock sync.Mutex
  10.      var maplist map[string]int
  11.      maplist = map[string]int{"one": 1, "two": 2}
  12.      lock.Lock()
  13.      maplist["three"] = 3
  14.      lock.Unlock()
  15.      fmt.Println(maplist)
  16. }

执行结果

Go语言中map使用和并发安全详解

(2)sync.Map包

  1. package main
  2.  
  3. import (
  4.      "fmt"
  5.      "sync"
  6. )
  7.  
  8. func main() {
  9.      m := sync.Map{} //或者 var mm sync.Map
  10.      m.Store("a", 1)
  11.      m.Store("b", 2)
  12.      m.Store("c", 3) //插入数据
  13.      fmt.Println(m.Load("a")) //读取数据
  14.      m.Range(func(key, value interface{}) bool { //遍历
  15.          fmt.Println(key, value)
  16.          return true
  17.      })
  18. }

执行结果

Go语言中map使用和并发安全详解

我们称其为并发安全的map。

 

总结

到此这篇关于Go语言中map使用和并发安全的文章就介绍到这了,更多相关Go语言map并发安全内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/weixin_44908159/article/details/123609779

延伸 · 阅读

精彩推荐
  • Golanggolang生成指定位数的随机数的方法

    golang生成指定位数的随机数的方法

    这篇文章主要介绍了golang生成指定位数的随机数的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友...

    TigerwolfC11972020-05-29
  • GolangGo语言基于Socket编写服务器端与客户端通信的实例

    Go语言基于Socket编写服务器端与客户端通信的实例

    这篇文章主要介绍了Go语言基于Socket编写服务器端与客户端通信的实例,包括实现基于自定义通讯协议的Socket通信,需要的朋友可以参考下 ...

    G1SLu3282020-08-13
  • GolangGo语言实现Base64、Base58编码与解码

    Go语言实现Base64、Base58编码与解码

    本文主要介绍了Base64、Base58编码与解码,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    小生凡一7842021-08-17
  • Golanggolang结构体与json格式串实例代码

    golang结构体与json格式串实例代码

    本文通过实例代码给大家介绍了golang结构体与json格式串的相关知识,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下 ...

    wilson935612020-05-21
  • Golanggolang 获取明天零点的时间戳示例

    golang 获取明天零点的时间戳示例

    今天小编就为大家分享一篇golang 获取明天零点的时间戳示例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    一名路过的小码农17462020-05-15
  • GolangGO语言中=和:=的区别说明

    GO语言中=和:=的区别说明

    这篇文章主要介绍了GO语言中=和:=的区别说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    持之以恒20166162021-02-27
  • Golang从 bug 中学习:六大开源项目告诉你 go 并发编程的那些坑

    从 bug 中学习:六大开源项目告诉你 go 并发编程的那些坑

    并发编程中,go不仅仅支持传统的通过共享内存的方式来通信,更推崇通过channel来传递消息,这种新的并发编程模型会出现不同于以往的bug。...

    腾讯技术8762021-02-24
  • GolangGo语言中的Iota关键字

    Go语言中的Iota关键字

    这篇文章介绍了Go语言中的Iota关键字,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...

    奋斗的大橙子9932022-07-16