设为首页 收藏本站
查看: 1105|回复: 0

[经验分享] Golang的锁和线程安全的Map

[复制链接]

尚未签到

发表于 2018-9-19 13:06:48 | 显示全部楼层 |阅读模式
  Golang的包sync实现了两种类型的锁: sync.Mutex 和 sync.RWMutex。通过阅读源代码我们可以知道sync.RWMutex是基于sync.Mutex实现的,其中的只读锁的实现使用类似引用计数的方式。
  对于任意 sync.Mutex 或 sync.RWMutex 变量l。 如果 n < m ,那么第n次 l.Unlock() 调用在第 m次 l.Lock()调用返回前发生。
  比如例子:
  package main
  import (
  "sync"
  )
  var l sync.Mutex
  var a string
  func f() {
  a = "hello, world"
  l.Unlock()
  }
  func main() {
  l.Lock()
  go f()
  l.Lock()
  print(a)
  }
  这个例子会输出 hello,world 的。
  因为,第一次 l.Unlock() 调用(在f函数中)在第二次 l.Lock() 调用(在main 函数中)返回之前发生,也就是在 print 函数调用之前发生。
  比如如果代码修改成下面方式,
  package main
  import (
  "sync"
  )
  var l sync.Mutex
  var a string
  func f() {
  a = "hello, world"
  print("21,")
  l.Unlock()
  print("22,")
  }
  func main() {
  l.Lock()
  print("1,")
  f()
  print("2,")
  l.Lock()
  print("3,")
  print(a)
  }
  执行结果是:1,21,22,2,3,hello, world
  如果代码改成下面方式
  package main
  import (
  "sync"
  )
  var l sync.Mutex
  var a string
  func f() {
  a = "hello, world"
  print("21,")
  l.Unlock()
  print("22,")
  }
  func main() {
  l.Lock()
  print("1,")
  go f()
  print("2,")
  l.Lock()
  print("3,")
  print(a)
  }
  执行结果是:1,2,21,22,3,hello, world
  下面代码来自:https://github.com/astaxie/beego/blob/master/safemap.go  用读写锁实现了 安全的map
  package beego
  import (
  "sync"
  )
  type BeeMap struct {
  lock *sync.RWMutex
  bm   map[interface{}]interface{}
  }
  func NewBeeMap() *BeeMap {
  return &BeeMap{
  lock: new(sync.RWMutex),
  bm:   make(map[interface{}]interface{}),
  }
  }
  //Get from maps return the k's value
  func (m *BeeMap) Get(k interface{}) interface{} {
  m.lock.RLock()
  defer m.lock.RUnlock()
  if val, ok := m.bm[k]; ok {
  return val
  }
  return nil
  }
  // Maps the given key and value. Returns false
  // if the key is already in the map and changes nothing.
  func (m *BeeMap) Set(k interface{}, v interface{}) bool {
  m.lock.Lock()
  defer m.lock.Unlock()
  if val, ok := m.bm[k]; !ok {
  m.bm[k] = v
  } else if val != v {
  m.bm[k] = v
  } else {
  returnfalse
  }
  return true
  }
  // Returns true if k is exist in the map.
  func (m *BeeMap) Check(k interface{}) bool {
  m.lock.RLock()
  defer m.lock.RUnlock()
  if _, ok := m.bm[k]; !ok {
  returnfalse
  }
  return true
  }
  func (m *BeeMap) Delete(k interface{}) {
  m.lock.Lock()
  defer m.lock.Unlock()
  delete(m.bm, k)
  }
  参考资料:
  go_mem   内存模型
  https://code.google.com/p/golang-china/wiki/go_mem
  http://hugozhu.myalert.info/2013/04/20/31-golang-memory-model.html
  Golang读写锁的实现源码
  http://golang.org/src/pkg/sync/rwmutex.go
  用读写锁实现的线程安全的Map
  https://github.com/astaxie/beego/blob/master/safemap.go


运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-594291-1-1.html 上篇帖子: Golang包管理工具glide简介 下篇帖子: golang 线程与通道
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表