Go中单例模式通过sync.Once实现线程安全延迟加载:用私有指针变量+Once.Do确保首次调用才初始化,避免DCL竞态;支持闭包传参、测试替换和资源释放。
在 Go 语言中实现单例模式并支持延迟加载(Lazy Initialization),关键在于**线程安全地确保实例仅在首次使用
时创建,且全局唯一**。Go 没有类和构造函数,但可通过包级变量 + `sync.Once` 高效、简洁地达成目标。
`sync.Once` 是 Go 标准库提供的工具,保证其 `Do` 方法内的函数只执行一次,天然适配单例的“首次调用才初始化”需求。这是最推荐、最轻量的方式。
常见误区是手动写“if instance == nil { lock; if nil { create } }”,即双重检查锁(DCL)。Go 中这不仅冗余,还容易因内存模型问题引入竞态。`sync.Once` 内部已做充分优化与内存屏障处理,直接用它更安全可靠。
标准 `sync.Once` 不接受参数,若单例初始化依赖外部输入(如配置路径、环境名),可将参数提前传入,或通过闭包捕获:
为便于单元测试(如替换 mock 实例),可提供一个 SetInstanceForTest 函数(仅在 test 文件中调用),临时覆盖私有 instance 变量。生产代码不受影响,又保测试灵活性。
基本上就这些。Go 的单例不靠语法限制,而靠约定 + 工具(Once) + 封装(私有变量 + 导出方法)来保障正确性。延迟加载自然融入其中,不复杂但容易忽略细节。