记一次golang中sync.Map并发创建、读取的问题

 cunfate

https://www.jianshu.com/p/f472e79909bc

背景:

我们有一个用go做的项目,其中用到了zmq4进行通信,一个简单的rpc过程,早期远端是使用一个map去做ip和具体socket的映射。

问题

大概是这样

struct SocketMap {sync.Mutexsockets map[string]*zmq4.Socket
}

然后调用的时候的代码大概就是这样的:

func (pushList *SocketMap) push(ip string, data []byte) {pushList.Lock()defer pushList.UnLock()socket := pushList.sockets[string]if socket == nil {socket := zmq4.NewSocket()//do some initial operation like connectpushList.sockets[ip] = socket}socket.Send(data)
}

相信大家都能看出问题:当push被并发访问的时候(事实上push会经常被并发访问),由于这把大锁的存在,同时只能有一个协程在临界区工作,效率是会被大大降低的。

解决方案:会带来crash的优化

所以我们决定使用sync.Map来替代这个设计,然后出了第一版代码,写的非常简单,只做了简单的替换:

struct SocketMap {sockets sync.Map
}func (pushList *SocketMap) push(ip string, data []byte) {var socket *zmq4.Socket    socketInter, ok = pushList.sockets.Load(ip)if !ok {socket = zmq4.NewSocket()//do some initial operation like connectpushList.sockets.Store(ip, socket)} else {socket = socketInter.(*zmq4.Socket)}socket.Send(data)
}

乍一看似乎没什么问题?但是跑起来总是爆炸,然后一看log,提示有个非法地址。后来在github上才看到,zmq4.Socket不是线程安全的。上面的代码恰恰会造成多个线程同时拿到socket实例,然后就crash了。

解决方案2: 加一把锁也挡不住的冲突

然后怎么办呢?看来也只能加锁了,不过这次加锁不能加到整个map上,否则还会有性能问题,那就考虑减小锁的粒度吧,使用锁包装socket。这个时候我们的代码也就呼之欲出了:

struct SocketMutex{sync.Mutexsocket *zmq4.Socket
}
struct SocketMap {sockets sync.Map
}func (pushList *SocketMap) push(ip string, data []byte) {var socket *SocketMutex    socketInter, ok = pushList.sockets.Load(ip)if !ok {socket = &{socket: zmq4.NewSocket()}//do some initial operation like connectpushList.sockets.Store(ip, newSocket)} else {socket = socketInter.(*SocketMutex)}socket.Lock()defer socket.Unlock()socket.socket.Send(data)
}

但是这样还是有问题,相信经验比较丰富的老哥一眼就能看出来,问题处在socketInter, ok = pushList.sockets.Load(ip)这行代码上,如果map中没有这个值,且有多个协程同时访问到这行代码,显然这几个协程的ok都会置为false,然后都进入第一个if代码块,创建多个socket实例,并且争相覆盖原有值。
单纯解决这个问题也很简单,就是使用sync.Map.LoadOrStore(key interface{}, value interface{}) (v interface{}, loaded bool)这个api,来原子地去做读写。
然而这还没完,我们的写入新值的操作不光是调用一个api创建socket就完了,还要有一系列的初始化操作,我们必须保证在初始化完成之前,其他通过Load拿到这个实例的协程无法真正访问socket实例。
这时候显然sync.Map自带的机制已经无法解决这个问题了,那么我们必须寻求其他的手段,要么锁,要么就sync.WaitGroup或者whatever的其他什么东西。

解决方案3: 闭包带来的神奇体验

后来经大佬指点,我在encoder.go中看到了这么一段代码:

 346 func typeEncoder(t reflect.Type) encoderFunc {                                 347     if fi, ok := encoderCache.Load(t); ok {                                     348         return fi.(encoderFunc)                                                349     }                                                                          350                                                                                351     // To deal with recursive types, populate the map with an                  352     // indirect func before we build it. This type waits on the                353     // real func (f) to be ready and then calls it. This indirect              354     // func is only used for recursive types.                                  355     var (                                                                      356         wg sync.WaitGroup                                                      357         f  encoderFunc                                                         358     )                                                                          359     wg.Add(1)                                                                  360     fi, loaded := encoderCache.LoadOrStore(t, encoderFunc(func(e *encodeState, v reflect.Value, opts encOpts) {361         wg.Wait()                                                              362         f(e, v, opts)                                                          363     }))                                                                        364     if loaded {                                                                365         return fi.(encoderFunc)                                                366     }                                                                          367                                                                                368     // Compute the real encoder and replace the indirect func with it.         369     f = newTypeEncoder(t, true)                                                370     wg.Done()                                                                  371     encoderCache.Store(t, f)                                                   372     return f                                                                   373 }

豁然开朗,我们可以在sync.Map中存放一个闭包函数,然后在闭包函数中等待本地的sync.WaitGroup完成再返回实例。于是最终的代码也就成型了。

struct SocketMutex{sync.Mutexsocket *zmq4.Socket
}
struct SocketMap {sockets sync.Map
}func (pushList *SocketMap) push(ip string, data []byte) {type SocketFunc func()*SocketMutexvar (socket *SocketMutexw sync.WaitGroup)socket = &SocketMutex {socket : zmq4.NewSocket()}    w.Add(1)socketf, ok = pushList.sockets.LoadOrStore(ip, SocketFunc(func()*SocketMutex) {w.Wait()return socket})if !ok {socket = &{socket: zmq4.NewSocket()}//do some initial operation like connectw.Done()} else {socket = socketInter.(*SockeFunc)()}socket.Lock()defer socket.Unlock()socket.socket.Send(data)
}

总结:

并发代码中的竞争问题,每一行代码的重入性都要深思熟虑啊。
总的来说要保持以下几个准则:

(1) 不可重入访问的系统资源,如socketfd, filefd,signalfd(事实上大多数这种系统资源都是不可重入的)等,在使用无锁结构的容器、读写锁封装的容器时,需要给每个资源单独加锁或者使用其他手段保证系统资源在临界区受到有效保护。

(2)如果有读取,如果为空则写入的逻辑,需要使用能提供原子性保证的LoadOrSave调用,或者没有的话,自己实现也要保证读取和写入过程整体的原子性;防止并发访问Load调用时,多个线程都返回否而创建多个实例,然后在Save的时候又互相覆盖。——这个原则不光对成员是系统资源的时候生效,如果存放的是其他东西也同样适用。

(3)如果资源创建完毕,还需要其他的初始化过程,则可以考虑在容器内放置闭包,初始化过程使用sync.WaitGroup保护,在闭包中调用Wait方法等待初始化完成再给其他线程返回初始化好的实例。而初始化过程完成后,可以置换闭包函数,不再调用Wait方法,来减少可能的开销。

记一次golang中sync.Map并发创建、读取的问题相关推荐

  1. map 值为指针_Go sync.Map 并发效率为什么那么高?

    点击上方蓝色"后端开发杂谈"关注我们, 专注于后端日常开发技术分享 Go sync.Map揭秘 简介: 对于熟悉 Go 语言的同学都知道, Go 自身的 map 是不支持并发读写, ...

  2. 由浅入深聊聊Golang的sync.Map

    前言 今天在技术群中有小伙伴讨论并发安全的东西,其实之前就有写过map相关文章:由浅入深聊聊Golang的map.但是没有详细说明sync.Map是怎么一回事. 回想了一下,竟然脑中只剩下" ...

  3. Golang中常用的并发模型

    通过channel通知实现并发控制 通过sync包中的WaitGroup实现并发控制,它会等待它收集的所有 goroutine 任务全部完成 在Go 1.7 以后引进的强大的Context上下文,实现 ...

  4. golang中的map

    前几天写代码,发现程序执行结果总是无规律的变动,最后一点点加日志查原因,才发现是map使用方式不对 1 程序中创建了一个类似如下的结构 mapOne := map[string]map[string] ...

  5. 1.13 golang中的Map

    1. Map map是一种无序的基于key-value的数据结构,Go语言中的map是引用类型,必须初始化才能使用. 1.1.1. map定义 Go语言中 map的定义语法如下 map[KeyType ...

  6. Golang 中 sync/atomic 包的原子操作

    背景 Go中多协程的情况下, 要保证操作的原子性,一般要使用RWMutex或者Mutex, 但是锁使用起来比较复杂,还要考虑lock 和unlock 顺序和成对出现,不注意就容易出错. 于是在sync ...

  7. Golang sync.Map 原理(两个map实现 读写分离、适用读多写少场景)

    参考: 由浅入深聊聊Golang的sync.Map 通过对源码的逐行分析,清晰易懂 Golang sync.Map原理 通过向 sync.Map 中增删改查来介绍sync.Map的底层原理 Golan ...

  8. Golang sync.Map 简介与用法

    Golang 中的 map 在并发情况下,只读是线程安全的,并发读写线程不安全.为了解决这个问题,Golang 提供了语言层级的并发读写安全的 sync.Map. type Map struct {/ ...

  9. Golang sync.Map原理

    原生map的"先天不足" 对于已经初始化了的原生map,我们可以尽情地对其进行并发读: package mainimport ("fmt""math/ ...

最新文章

  1. 解决“由于应用程序的配置不正确,应用程序未能启动,重新安装应用程序可能会纠正这个问题”(转贴)...
  2. [转]代码分析工具FxCop1.36之一:介绍与使用
  3. 实战部署MySQL用户认证的Postfix邮件系统(3)
  4. element 保存 点击链接_保存微信表情!
  5. [转]如何用VS2005制作Web安装程序
  6. 解决bert在文本长度的问题的方案,篇章级
  7. 只属于你我的共同记忆
  8. fastboot工具的操作流程
  9. 2020.9.28 爱客影视3.6.5-pro完整版搭建完毕带解析接口无广告
  10. 科研萌新成长记8——我不是不接受上帝,我只是不接受上帝创造的这个世界
  11. 华为和荣耀的关系:亲兄弟,各自为王
  12. dpdk pmd驱动初始化
  13. Java编程基础之Set和Map的简单使用
  14. linux上下左右为ABCD
  15. 后台管理系统--首页及登录认证
  16. haproxy中的Proxies段的配置
  17. qq邮箱服务器接收和发送文件夹,将QQ邮箱打造成为你的邮箱总管-qq邮箱怎么发送文件夹...
  18. 算法-动态规划学习(含经典例子分析)
  19. 总结moba游戏与皇室战争类游戏帧同步区别
  20. 亚马逊云计算免费一年被扣款

热门文章

  1. 麦克纳姆轮全向移动机器人运动效率分析
  2. 单线程为什么cpu慢_面试时说Redis是单线程的,被喷惨了!
  3. 洛谷——P1089 [NOIP2004 提高组] 津津的储蓄计划
  4. 转:Nginx 性能优化有这篇就够了!
  5. 从零搭建自己的SpringBoot后台框架(六)
  6. python中关于字典的基础运用
  7. python字符串isalnum()函数
  8. JAVA大数据习题Week2
  9. PHP设计渐变的效果,canvas渐变色:canvas如何实现渐变色的效果?
  10. 【李宏毅2020 ML/DL】P110-111 Policy Gradient Proximal Policy Optimization