Go语言中的sync.Map怎么实现
偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《Go语言中的sync.Map怎么实现》,这篇文章主要会讲到sync.Map、go语言等等知识点,不知道大家对其都有多少了解,下面我们就一起来看一吧!当然,非常希望大家能多多评论,给出合理的建议,我们一起学习,一起进步!
这篇“Go语言中的sync.Map怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Go语言中的sync.Map怎么实现”文章吧。
1. Map 的基本实现原理
在介绍 sync.Map 的基本实现原理之前,我们需要先了解一下 Go 语言标准库中的 map 实现原理。在 Go 中,map 是基于哈希表实现的。当我们向 map 中添加元素时,它会根据 key 计算出一个哈希值,然后将这个值映射到一个桶中。如果该桶中已经有了元素,它会遍历桶中的元素,查找是否已经存在相同的 key,如果存在就更新对应的值,否则就添加一个新的键值对。
下面是一个简单的 map 示例:
m := make(map[string]int) m["a"] = 1 m["b"] = 2 fmt.Println(m["a"]) // Output: 1
当我们运行这段代码时,Go 语言会自动帮我们分配一个哈希表和若干个桶,然后将键值对添加到对应的桶中。这样,当我们需要访问某个 key 对应的值时,Go 语言会根据哈希值快速定位到对应的桶,然后遍历桶中的元素,查找是否有相同的 key,如果找到了就返回对应的值。
2. sync.Map 的实现原理
sync.Map 是 Go 语言标准库中的一个并发安全的 Map 实现,它可以在并发情况下安全地读写,而不需要加锁。那么,它是如何实现这种并发安全性的呢?下面我们就来一步步地解析 sync.Map 的实现原理。
2.1 sync.Map 的结构体定义
首先,让我们来看一下 sync.Map 的结构体定义:
type Map struct { mu sync.Mutex read atomic.Value // readOnly dirty map[interface{}]interface{} misses int dirtyLocked uintptr }
从上面的代码中可以看出,sync.Map 的实现主要是依赖于一个互斥锁(sync.Mutex)和两个 map(read 和 dirty)。其中,read 和 dirty 的作用分别是什么呢?我们先来看一下 read 的定义:
type readOnly struct { m map[interface{}]interface{} amended bool }
可以看到,read 只有一个成员 m,它是一个 map 类型。而 amended 则表示 read 中的键值对是否被修改过。接下来,我们来看一下 dirty 的定义:
type dirty struct { m map[interface{}]interface{} dirty map[interface{}]bool misses int }
和 read 不同的是,dirty 中包含了两个 map:m 和 dirty。其中,m 存储了被修改过的键值对,而 dirty 则存储了哪些键值对被修改过。
2.2 sync.Map 的读取实现
在 sync.Map 中,读取操作非常简单,直接从 readOnly 中的 m 中查找即可。如果 readOnly 中的键值对被修改过,则需要从 dirty 中查找。读取操作的实现代码如下:
func (m *Map) Load(key interface{}) (value interface{}, ok bool) { read, _ := m.read.Load().(readOnly) value, ok = read.m[key] if !ok && read.amended { m.mu.Lock() read, _ = m.read.Load().(readOnly) value, ok = read.m[key] if !ok && read.amended { value, ok = read.m[key] } m.mu.Unlock() } return }
在这段代码中,我们首先从 readOnly 中的 m 中查找键值对。如果键值对不存在且 readOnly 中的键值对被修改过,则需要获取互斥锁,并重新从 readOnly 中查找。如果还是没有找到,那么就从 dirty 中查找。
2.3 sync.Map 的写入实现
在 sync.Map 中,写入操作需要分两步完成。首先,我们需要判断 readOnly 中的键值对是否被修改过,如果没有被修改过,则直接将键值对添加到 readOnly 中的 m 中即可。否则,我们需要获取互斥锁,然后将键值对添加到 dirty 中的 m 中,并将对应的键添加到 dirty 中的 dirty 中。写入操作的实现代码如下:
func (m *Map) Store(key, value interface{}) { read, _ := m.read.Load().(readOnly) if v, ok := read.m[key]; !ok && !read.amended { m.mu.Lock() read, _ = m.read.Load().(readOnly) if v, ok := read.m[key]; !ok { read = readOnly{m: read.m, amended: true} } read.m[key] = value m.read.Store(read) m.mu.Unlock() } else { m.mu.Lock() dirty := m.dirtyLocked != 0 if !dirty { m.dirtyLocked = 1 m.dirty = make(map[interface{}]interface{}) } m.dirty[key] = value if !ok { m.dirty[key] = value m.dirty[key] = true } if dirty { m.mu.Unlock() return } m.read.Store(readOnly{m: read.m, amended: true}) m.mu.Unlock() } }
在这段代码中,我们首先从 readOnly 中的 m 中查找键值对。如果键值对不存在且 readOnly 中的键值对没有被修改过,则需要获取互斥锁,并重新从 readOnly 中查找。如果还是没有找到,则将键值对添加到 readOnly 中的 m 中,并将 amended 设置为 true。否则,我们需要获取互斥锁,并将键值对添加到 dirty 中的 m 中,并将对应的键添加到 dirty 中的 dirty 中。如果 dirty 中已经存在该键,则只需要更新 dirty 中的键值即可。如果 dirty 中没有该键,则需要在 dirty 中添加该键,并将该键的 dirty 置为 true。
接下来,我们需要判断 dirty 是否被锁定。如果 dirty 被锁定,则直接退出函数。否则,我们需要将 readOnly 中的 amended 设置为 true,并将 readOnly 存储回 read 中。
2.4 sync.Map 的删除实现
在 sync.Map 中,删除操作也需要分两步完成。首先,我们需要判断 readOnly 中的键值对是否被修改过,如果没有被修改过,则直接从 readOnly 中的 m 中删除键值对即可。否则,我们需要获取互斥锁,然后将键添加到 dirty 中的 dirty 中,并将 dirty 中的对应键的值设置为 false。删除操作的实现代码如下:
func (m *Map) Delete(key interface{}) { read, _ := m.read.Load().(readOnly) if _, ok := read.m[key]; ok || read.amended { m.mu.Lock() read, _ = m.read.Load().(readOnly) if _, ok := read.m[key]; ok || read.amended { if m.dirty == nil { m.dirty = make(map[interface{}]interface{}) } m.dirty[key] = false m.dirty[key] = true m.read.Store(readOnly{m: read.m, amended: true}) } m.mu.Unlock() } }
在这段代码中,我们首先从 readOnly 中的 m 中查找键值对。如果键值对存在或者 readOnly 中的键值对被修改过,则需要获取互斥锁,并重新从 readOnly 中查找。如果还是没有找到,则将键添加到 dirty 中的 dirty 中,并将 dirty 中的对应键的值设置为 false。接下来,我们需要判断 dirty 是否为 nil,如果为 nil,则需要将 dirty 初始化为一个空 map。然后,我们将键添加到 dirty 中,并将 dirty 中的对应键的值设置为 true。最后,我们将 readOnly 中的 amended 设置为 true,并将 readOnly 存储回 read 中。
2.5 sync.Map 的遍历实现
在 sync.Map 中,遍历操作需要将 readOnly 和 dirty 中的所有键值对进行合并,并返回所有未被删除的键值对。遍历操作的实现代码如下:
func (m *Map) Range(f func(key, value interface{}) bool) { read, _ := m.read.Load().(readOnly) if read.amended { m.mu.Lock() read, _ = m.read.Load().(readOnly) if read.amended { read = readOnly{ m: merge(read.m, m.dirty), } read.amended = false m.read.Store(read) m.dirty = nil } m.mu.Unlock() } for k, v := range read.m { if !f(k, v) { break } } } func merge(m1, m2 map[interface{}]interface{}) map[interface{}]interface{} { if len(m1) == 0 && len(m2) == 0 { return nil } if len(m1) == 0 { return m2 } if len(m2) == 0 { return m1 } m := make(map[interface{}]interface{}) for k, v := range m1 { m[k] = v } for k, v := range m2 { if _, ok := m[k]; !ok || !v.(bool) { m[k] = v } } return m }
在这段代码中,我们首先从 readOnly 中获取所有的键值对,并检查是否有键值对被修改过。如果键值对被修改过,则需要获取互斥锁,并将 readOnly 和 dirty 中的键值对合并,然后将合并后的键值对存储回 readOnly 中,并将 dirty 设置为 nil。接下来,我们遍历 readOnly 中的所有键值对,并调用 f 函数来处理键值对。如果 f 函数返回 false,则遍历过程结束。
在这个 Range 函数中,我们还实现了一个名为 merge 的辅助函数,用于合并两个 map。在合并过程中,我们首先判断两个 map 是否为空,如果为空,则直接返回 nil。如果其中一个 map 为空,则返回另一个 map。否则,我们需要将 m1 中的键值对全部添加到新的 map 中,并逐个遍历 m2 中的键值对。如果 m2 中的键不存在于新的 map 中,或者 m2 中的键被删除,则将其添加到新的 map 中。
以上就是关于“Go语言中的sync.Map怎么实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注golang学习网行业资讯频道。
今天带大家了解了sync.Map、go语言的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

- 上一篇
- DAMO-YOLO:兼顾速度与精度的高效目标检测框架

- 下一篇
- golang的websocket服务端识别客户端
-
- Golang · Go教程 | 13分钟前 |
- 值类型传递特点,函数拷贝影响详解
- 361浏览 收藏
-
- Golang · Go教程 | 14分钟前 |
- Golang空接口interface{}用法与类型断言详解
- 449浏览 收藏
-
- Golang · Go教程 | 45分钟前 |
- Golang基准测试编写与性能优化指南
- 385浏览 收藏
-
- Golang · Go教程 | 50分钟前 |
- GoogleAppEngineGoSDK模板问题解决方法
- 261浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang插件中反射的应用与实现详解
- 489浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- GolangHTTP优化:Keep-Alive与连接复用设置
- 211浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang排序技巧:sort包与自定义对比解析
- 244浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang协程同步:WaitGroup使用全解析
- 251浏览 收藏
-
- Golang · Go教程 | 1小时前 | 可读性 Go测试 断言函数 testify/assert testify/require
- Golang断言库assert使用教程
- 104浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- 高性能原型语言:易用快速之选
- 418浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 420次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 419次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 415次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 433次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 450次使用
-
- 有关Go语言拼接URL路径的方法
- 2023-03-09 185浏览
-
- go语言能不能做后端
- 2023-03-03 460浏览
-
- go语言和java的区别是什么
- 2023-03-03 430浏览
-
- go语言如何进行强制类型转换
- 2023-03-04 450浏览
-
- go语言的beego怎么使用
- 2023-03-03 320浏览