Golang sync.Once

  1. 简介

    Go 标准库提供的使函数只执行一次的实现,常应用于单例模式,例如初始化配置、保持数据库连接等

  2. 与init区别

    • init 函数是当所在的 package 首次被加载时执行
    • sync.Once 可以在代码的任意位置初始化和调用,可以延迟到使用时再执行,且并发场景下是线程安全的
  3. 多数情况下,sync.Once 被用于控制变量的初始化,这个变量的读写满足如下三个条件

    • 当且仅当第一次访问某个变量时,进行初始化(写)
    • 变量初始化过程中,所有读都被阻塞,直到初始化完成
    • 变量仅初始化一次,初始化完成后驻留在内存里
  4. 原理

    • 保证变量仅被初始化一次,需要有个标志来判断变量是否已初始化过,若没有则需要初始化
    • 线程安全,支持并发,无疑需要互斥锁来实现
    • 源码实现
    package sync
    
    import (
        "sync/atomic"
    )
    
    type Once struct {
        done uint32
        m    Mutex
    }
    
    func (o *Once) Do(f func()) {
        if atomic.LoadUint32(&o.done) == 0 {
            o.doSlow(f)
        }
    }
    
    func (o *Once) doSlow(f func()) {
        o.m.Lock()
        defer o.m.Unlock()
        if o.done == 0 {
            defer atomic.StoreUint32(&o.done, 1)
            f()
        }
    }
    
    • 为什么放在第一个字段就能够减少指令呢?因为结构体第一个字段的地址和结构体的指针是相同的,如果是第一个字段,直接对结构体的指针解引用即可。如果是其他的字段,除了结构体指针外,还需要计算与第一个值的偏移(calculate offset)。在机器码中,偏移量是随指令传递的附加值,CPU 需要做一次偏移值与指针的加法运算,才能获取要访问的值的地址。因为,访问第一个字段的机器代码更紧凑,速度更快。
  5. demo

    package main
    
    import (
    	"log"
    	"os"
    	"strconv"
    	"sync"
    	"time"
    )
    
    type Config struct {
    	Server string
    	Port   int64
    }
    
    var (
    	once   sync.Once
    	config *Config
    )
    
    func ReadConfig() *Config {
    	once.Do(func() {
    		var err error
    		config = &Config{Server: os.Getenv("TT_SERVER_URL")}
    		config.Port, err = strconv.ParseInt(os.Getenv("TT_PORT"), 10, 0)
    		if err != nil {
    			config.Port = 8080 // default port
    		}
    		log.Println("init config")
    	})
    	return config
    }
    
    func main() {
    	for i := 0; i < 10; i++ {
    		go func() {
    			_ = ReadConfig()
    		}()
    	}
    	time.Sleep(time.Second)
    }
    
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值