共计 1448 个字符,预计需要花费 4 分钟才能阅读完成。
导读 | 这篇文章主要介绍了 golang 实现单例的操作代码, 本文介绍基于 sync.Once 的方式来实现单例,熟练掌握这种模式,并理解其底层原理,对大部分人来讲已经完全够用了,需要的朋友可以参考下 |
在 go 里实现单例模式有多种方式:
本文介绍基于 sync.Once 的方式来实现单例,熟练掌握这种模式,并理解其底层原理,对大部分人来讲已经完全够用了。
基于 sync.Once 实现单例
// 其他 package 也可见,在其他地方也可以 new 新对象 | |
// 但是最终调用 Conn() 方法时,都是用的 single 这个单例 | |
// 1 | |
type Driver struct { | |
// 小写字母开头,外部不可访问,所以 new 个 Driver 新对象也没用 | |
// 2 | |
conn string | |
} | |
// 全局变量,指针默认值为 nil | |
// 3 | |
var single *Driver // 单例 | |
var once sync.Once | |
// 对外暴露的公共方法 | |
func (s *Driver) Conn() {fmt.Printf("conn=%s", single.conn) // do something | |
} | |
// 4 | |
func GetDriverSingleton() *Driver {// 对 GetDriverSingleton() 方法的调用,都会执行 once.Do() 方法,只不过参数 func() 只会被执行一次 | |
// 若并发执行 once.Do(),多个协程会阻塞,因内部是通过 Mutex 来控制 | |
once.Do(func() {single = new(Driver) | |
single.conn = "single conn" | |
time.Sleep(50 * time.Millisecond) | |
}) | |
return single | |
} |
单例类型定义 Driver
Driver 类的方法要支持跨包访问,因此需要以大写字母开头。
小写字母开头,作用域仅限于包内部。
类 Field conn
类变量 conn 需要小写字母开头,跨包不可访问,避免在包外被修改。
但是包内还是有可能被修改。
once.Do(func() {})
每次调用 GetDriverSingleton(),都会调用 once.Do() 方法,但是在 once.Do() 方法内部,仅会执行一次参数 func(){},因此就保证了单例唯一初始化。
并发访问 once.Do()
不会有并发访问问题,因 once.Do() 内部通过 mutex 来控制。
// once.DO() | |
if atomic.LoadUint32(&o.done) == 0 { | |
// Outlined slow-path to allow inlining of the fast-path. | |
o.doSlow(f) | |
} | |
// doSlow() | |
func (o *Once) doSlow(f func()) {o.m.Lock() // 互斥锁 | |
defer o.m.Unlock() | |
if o.done == 0 {defer atomic.StoreUint32(&o.done, 1) | |
f()} | |
} |
对外暴露方法 Conn()
外部对 Conn() 方法的调用,最终都由单例 single 来实现。
重新 new(Driver) 会发生什么?
很遗憾,无法将构造函数改成 private,也就是说,在包外部是可以通过 new(Driver) 来创建新的对象。
但无论是哪个对象,对公开方法 Conn() 的调用,最终都是由单例 single 来执行的。
到此这篇关于 golang 实现单例的操作代码的文章就介绍到这了
正文完
星哥玩云-微信公众号
