Go语言sync.Map平行宇宙入门指南

简介: Go语言sync.Map平行宇宙入门指南

/ Go 语言 sync.Map 用法详解 /

Go 语言中的 map 在并发操作时不是线程安全的,为了解决这个问题,Go 1.9 版本添加了 sync.Map 这种并发安全的 map 实现。正确使用 sync.Map 可以有效防止并发环境下的数据竞争问题。

本文将详细介绍 sync.Map 的相关用法,内容涵盖:

  1. sync.Map 背景
  2. sync.Map 接口介绍
  3. sync.Map 的存储读取删除
  4. sync.Map 遍历
  5. sync.Map 与普通 map 的区别
  6. sync.Map 使用注意事项
  7. sync.Map 应用示例
  8. sync.Map 原理简析

通过本文可以全面了解 sync.Map 的工作原理及使用技巧,用来构建高效的并发程序。


1

 

1. sync.Map 背景

Go 语言中的 map 在多线程并发使用时不是线程安全的,主要问题是读写 map 时会存在数据竞争,导致错误。

例如并发读写 map:

var m = map[string]int{}
go func() {
  for { 
    m["count"]++
  }
}()
go func() {
  for {
    _ = m["count"] 
  }  
}()

这样会引发 fatal error: concurrent map read and map write。

为了解决这个问题,在 Go 1.9 中添加了 sync.Map 做为并发安全的 map 实现。


2

 

2. sync.Map 接口

sync.Map 的接口实现了一个自动并发安全的 map,主要方法有:

func New() *Map   // 创建一个空的Map
m.Store(key, value) // 存储key-value
m.Load(key) // 获取key对应的value
m.LoadOrStore(key, value) // 获取或存储并返回值
m.Delete(key) // 删除key 
m.Range(f func(key, value interface{})) // 范围遍历

这些方法都天生具备并发安全访问能力,使用 sync.Map 就不会有并发数据竞争的风险。

我们可以直接通过 sync.Map 来并发安全地使用一个 map:

var m sync.Map
m.Store("count", 0)
// 并发安全地读写map

3

 

3. 存储和读取

sync.Map 提供了 Store 和 Load 方法进行数据存储和读取:

m.Store(key, value) // 存储 
value, ok := m.Load(key) // 读取
if !ok {
  // key不存在
} else {
  // 获取value
}

Store 会将 key-value 放入 sync.Map 中存储。

Load 方法可以读取 key 对应的 value,它返回值和一个 bool 类型表示是否读取成功。

同时提供了 LoadOrStore 方法,可以读取或存储:

actual, loaded = m.LoadOrStore(key, value)

它可以获取 key 对应的 value,如果 key 不存在就保存这个 value 并返回。


4

 

4. 删除元素

可以使用 Delete 方法来删除元素:

m.Delete(key)

删除成功后,再 Load 这个 key 会返回 value 类型的默认零值。

和普通 map 类似,删除不存在的 key 不会报错。


5

 

5. 遍历

sync.Map 提供了 Range 方法用于遍历所有键值对:

m.Range(func(k, v interface{}) bool {
  // 在此处理 kv
  return true
})

遍历是无序的,和 map 的迭代顺序一致。

返回 true 会继续迭代,false 会停止遍历。


6

 

6. 与普通 map 区别

sync.Map 和普通的 map 有以下区别:

  • sync.Map 设计为并发安全,不需要额外同步
  • 没有容量概念,扩容自动处理
  • 只提供了常用的 map 操作方法
  • 不保证遍历顺序
  • 没有直接转换为普通 map 的方法

在需要并发安全时选择 sync.Map,其他情况普通 map 更加高效。


7

 

7. 使用注意事项

使用 sync.Map 有以下注意事项:

  • 指针值作为 key 需要小心
  • 同时读写 key 会产生不确定行为
  • 删除时内存不能及时释放
  • 遍历时需要复制值进行处理

需要针对这些特点合理安全地使用。


8

 

8. 应用示例

sync.Map 常见使用场景:

  • 缓存
  • 计数器
  • 生产消费队列等

实现一个简单计数器:

var counter = sync.Map{}
func inc(key string) {
  counter.Store(key, 0)
  v := counter.Load(key)
  counter.Store(key, v+1)
} 
// 并发安全计数

9

 

9. sync.Map 原理简析

sync.Map 内部通过锁分段实现并发安全:

它使用一个包含多把锁的结构,每一把锁锁定一个 map 元素区间。

当访问元素时找到锁定的段,对该段加锁来保证安全:

segment1 map[1]     segment2 map[2]    segment3 map[3]
 |                 |                  |
lock1             lock2              lock3

这样实现多段锁,既保证了线程安全,又提高了效率。


10

 

总结

通过这篇文章,我们全面介绍了 sync.Map 的工作原理和使用技巧,它可以有效解决 Go 语言中 map 并发安全的问题。但也需要注意它和普通 map 的区别,合理使用。


目录
相关文章
|
4天前
|
存储 监控 算法
探秘员工泄密行为防线:基于Go语言的布隆过滤器算法解析
在信息爆炸时代,员工泄密行为对企业构成重大威胁。本文聚焦布隆过滤器(Bloom Filter)这一高效数据结构,结合Go语言实现算法,帮助企业识别和预防泄密风险。通过构建正常操作“指纹库”,实时监测员工操作,快速筛查可疑行为。示例代码展示了如何利用布隆过滤器检测异常操作,并提出优化建议,如调整参数、结合日志分析系统等,全方位筑牢企业信息安全防线,守护核心竞争力。
|
6天前
|
监控 Linux PHP
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
54 20
|
11天前
|
监控 关系型数据库 MySQL
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
22 0
|
12天前
|
Go C语言
Go语言入门:分支结构
本文介绍了Go语言中的条件语句,包括`if...else`、`if...else if`和`switch`结构,并通过多个练习详细解释了它们的用法。`if...else`用于简单的条件判断;`if...else if`处理多条件分支;`switch`则适用于基于不同值的选择逻辑。特别地,文章还介绍了`fallthrough`关键字,用于优化重复代码。通过实例如判断年龄、奇偶数、公交乘车及成绩等级等,帮助读者更好地理解和应用这些结构。
34 14
|
26天前
|
存储 监控 算法
内网监控系统之 Go 语言布隆过滤器算法深度剖析
在数字化时代,内网监控系统对企业和组织的信息安全至关重要。布隆过滤器(Bloom Filter)作为一种高效的数据结构,能够快速判断元素是否存在于集合中,适用于内网监控中的恶意IP和违规域名筛选。本文介绍其原理、优势及Go语言实现,提升系统性能与响应速度,保障信息安全。
29 5
|
1月前
|
存储 安全 Go
Go语言中的map数据结构是如何实现的?
Go 语言中的 `map` 是基于哈希表实现的键值对数据结构,支持快速查找、插入和删除操作。其原理涉及哈希函数、桶(Bucket)、动态扩容和哈希冲突处理等关键机制,平均时间复杂度为 O(1)。为了确保线程安全,Go 提供了 `sync.Map` 类型,通过分段锁实现并发访问的安全性。示例代码展示了如何使用自定义结构体和切片模拟 `map` 功能,以及如何使用 `sync.Map` 进行线程安全的操作。
【Go语言入门100题】026 I Love GPLT (5 分) Go语言 | Golang
L1-026 I Love GPLT (5 分) Go语言|Golang 这道超级简单的题目没有任何输入。 你只需要把这句很重要的话 —— “I Love GPLT”——竖着输出就可以了。 所谓“竖着输出”,是指每个字符占一行(包括空格),即每行只能有1个字符和回车。
621 0
|
存储 Go
【Go语言入门100题】023 输出GPLT (20 分) Go语言 | Golang
L1-023 输出GPLT (20 分) Go语言|Golang 给定一个长度不超过10000的、仅由英文字母构成的字符串。请将字符重新调整顺序,按GPLTGPLT....这样的顺序输出,并忽略其它字符。当然,四种字符(不区分大小写)的个数不一定是一样多的,若某种字符已经输出完,则余下的字符仍按GPLT的顺序打印,直到所有字符都被输出。 下面给出甲、乙两人的酒量(最多能喝多少杯不倒)和划拳记录,请你判断两个人谁先倒。
182 0
|
1月前
|
算法 安全 Go
Go语言中的加密和解密是如何实现的?
Go语言通过标准库中的`crypto`包提供丰富的加密和解密功能,包括对称加密(如AES)、非对称加密(如RSA、ECDSA)及散列函数(如SHA256)。`encoding/base64`包则用于Base64编码与解码。开发者可根据需求选择合适的算法和密钥,使用这些包进行加密操作。示例代码展示了如何使用`crypto/aes`包实现对称加密。加密和解密操作涉及敏感数据处理,需格外注意安全性。
48 14
|
1月前
|
Go 数据库
Go语言中的包(package)是如何组织的?
在Go语言中,包是代码组织和管理的基本单元,用于集合相关函数、类型和变量,便于复用和维护。包通过目录结构、文件命名、初始化函数(`init`)及导出规则来管理命名空间和依赖关系。合理的包组织能提高代码的可读性、可维护性和可复用性,减少耦合度。例如,`stringutils`包提供字符串处理函数,主程序导入使用这些函数,使代码结构清晰易懂。
98 11

热门文章

最新文章