sync.Once可以实现单例模式,确保sync.Once.Do(f func())只会被执行一次,可以初始化某个实例单例。

针对Golang 1.9的sync.Once,与Golang 1.10一样。 源代码位置:sync\once.go。

结构体

Once结构体定义如下:

type Once struct {
	m    Mutex
	done uint32   // 初始值为0表示还未执行过,1表示已经执行过
}

Do

func (o *Once) Do(f func()) {
    // done==1表示已经执行过了,直接结束返回
	if atomic.LoadUint32(&o.done) == 1 {
		return
	}
	// 锁住对象,避免并发问题
	o.m.Lock()
	defer o.m.Unlock()
	if o.done == 0 {
	    // 先将done设置为1,再执行f函数
		defer atomic.StoreUint32(&o.done, 1)
		f()
	}
}

需要注意的是执行f函数是同步进行的,也就是说可能存在阻塞问题。