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函数是同步进行的,也就是说可能存在阻塞问题。