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 的区别,合理使用。


目录
相关文章
|
2天前
|
安全 Java Go
探索Go语言在高并发环境中的优势
在当今的技术环境中,高并发处理能力成为评估编程语言性能的关键因素之一。Go语言(Golang),作为Google开发的一种编程语言,以其独特的并发处理模型和高效的性能赢得了广泛关注。本文将深入探讨Go语言在高并发环境中的优势,尤其是其goroutine和channel机制如何简化并发编程,提升系统的响应速度和稳定性。通过具体的案例分析和性能对比,本文揭示了Go语言在实际应用中的高效性,并为开发者在选择合适技术栈时提供参考。
|
6天前
|
运维 Kubernetes Go
"解锁K8s二开新姿势!client-go:你不可不知的Go语言神器,让Kubernetes集群管理如虎添翼,秒变运维大神!"
【8月更文挑战第14天】随着云原生技术的发展,Kubernetes (K8s) 成为容器编排的首选。client-go作为K8s的官方Go语言客户端库,通过封装RESTful API,使开发者能便捷地管理集群资源,如Pods和服务。本文介绍client-go基本概念、使用方法及自定义操作。涵盖ClientSet、DynamicClient等客户端实现,以及lister、informer等组件,通过示例展示如何列出集群中的所有Pods。client-go的强大功能助力高效开发和运维。
27 1
|
6天前
|
存储 安全 NoSQL
Go map 读写性能优化 - 分片 map
Go map 读写性能优化 - 分片 map
13 1
|
1天前
|
监控 NoSQL Go
Go语言中高效使用Redis的Pipeline
Redis 是构建高性能应用时常用的内存数据库,通过其 Pipeline 和 Watch 机制可批量执行命令并确保数据安全性。Pipeline 类似于超市购物一次性结账,减少网络交互时间,提升效率。Go 语言示例展示了如何使用 Pipeline 和 Pipelined 方法简化代码,并通过 TxPipeline 保证操作原子性。Watch 机制则通过监控键变化实现乐观锁,防止并发问题导致的数据不一致。这些机制简化了开发流程,提高了应用程序的性能和可靠性。
7 0
|
4天前
|
NoSQL Go Redis
Go语言中如何扫描Redis中大量的key
在Redis中,遍历大量键时直接使用`KEYS`命令会导致性能瓶颈,因为它会一次性返回所有匹配的键,可能阻塞Redis并影响服务稳定性。为解决此问题,Redis提供了`SCAN`命令来分批迭代键,避免一次性加载过多数据。本文通过两个Go语言示例演示如何使用`SCAN`命令:第一个示例展示了基本的手动迭代方式;第二个示例则利用`Iterator`简化迭代过程。这两种方法均有效地避免了`KEYS`命令的性能问题,并提高了遍历Redis键的效率。
15 0
|
5天前
|
监控 Serverless Go
Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
|
5天前
|
关系型数据库 MySQL 数据库连接
Go语言中使用sqlx来操作事务
在应用中,数据库事务保证操作的ACID特性至关重要。`github.com/jmoiron/sqlx`简化了数据库操作。首先安装SQLX和MySQL驱动:`go get github.com/jmoiron/sqlx`和`go get github.com/go-sql-driver/mysql`。导入所需的包后,创建数据库连接并使用`Beginx()`方法开始事务。通过`tx.Commit()`提交或`tx.Rollback()`回滚事务以确保数据一致性和完整性。
8 0
|
6天前
|
存储 算法 Go
go map 设计与实现
go map 设计与实现
9 0
|
9天前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
6天前
|
SQL 关系型数据库 MySQL
Go语言中使用 sqlx 来操作 MySQL
Go语言因其高效的性能和简洁的语法而受到开发者们的欢迎。在开发过程中,数据库操作不可或缺。虽然Go的标准库提供了`database/sql`包支持数据库操作,但使用起来稍显复杂。为此,`sqlx`应运而生,作为`database/sql`的扩展库,它简化了许多常见的数据库任务。本文介绍如何使用`sqlx`包操作MySQL数据库,包括安装所需的包、连接数据库、创建表、插入/查询/更新/删除数据等操作,并展示了如何利用命名参数来进一步简化代码。通过`sqlx`,开发者可以更加高效且简洁地完成数据库交互任务。
15 1