搜索
简帛阁>技术文章>Golang基于sync.Once实现单例的操作代码

Golang基于sync.Once实现单例的操作代码

在go里实现单例模式有多种方式:

  • 基于lock
  • 基于init函数
  • 基于sync.Once

本文介绍基于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来执行的。

目录基于syncOnce实现单例单例类型定义Driver类FieldconnonceDo(func(){})并发访问onceDo()对外暴露方法Conn()重新new(Driver)会发生什么?在go
录前言1定位2对外接口3实战用法31初始化32模式33关闭channel4原理5避坑前言在此前一篇文章中我们了解了GolangMutex原理解析,今天来看一个官方给出Mutex应用场景:syn
syncOnce常应用于模式,例如初始化配置、保持数据库连接等。init函数通常是所在package首次被加载时执行,如果一直没有被调用就会浪费内存。syncOnce可以在代码任意位置初始化和调用
超超为了能让婷婷过上幸福美满生活,决定去大厂历练一番,下面是他去大厂面试时遇到几个关于问题。一、认识单例超超:您好,面试官~面试官:你好,你平时开发是用windows还是linux居多?超超
syncWaitGroup,顾名思义,等待一组goroutinue运行完毕。syncWaitGroup声明后即可使用,它有如下方法:func(wg*WaitGroup)Add(deltaint)不能传
原文链接:https://blog.thinkeridea.com/202101/go/exsync/once.html官方描述Onceisanobjectthatwillperformexactly
yncOnce用于保证某个动作只被执行一次,可用于模式中,比如初始化配置。我们知道init()函数也只会执行一次,不过它是在main()函数之前执行,如果想要在代码执行过程中只运行某个动作一次,
一序单从库名大概就能猜出其作用。syncOnce使用起来很简单,下面是一个简单使用案例packagemainimport(fmtsync)funcmain(){var(oncesyncOncewg
syncOnce是Go标准库提供使函数只执行一次实现。作用与init函数类似,但有区别。在某些情况下预先初始化一个变量会增加函数启动延迟,如果实际执行时可能用不上这个变量,那么初始化就是非必须
syncwaitgroup功能WaitGroup使用多线程时,进行等待多线程执行完毕后,才可以结束函数,有两个选择channelwaitgroup首先使用channelfuncadd(n*int,