Golang单例与工厂模式怎么实现
时间:2026-01-02 11:21:35 311浏览 收藏
编程并不是一个机械性的工作,而是需要有思考,有创新的工作,语法是固定的,但解决问题的思路则是依靠人的思维,这就需要我们坚持学习和更新自己的知识。今天golang学习网就整理分享《Golang单例与工厂模式实现方法》,文章讲解的知识点主要包括,如果你对Golang方面的知识点感兴趣,就不要错过golang学习网,在这可以对大家的知识积累有所帮助,助力开发能力的提升。
Go中单例+工厂模式通过sync.Once实现线程安全单例,接口+工厂函数解耦实现,支持运行时配置与延迟初始化,避免init()硬编码、导出变量等陷阱。

在 Go 语言中,单例 + 工厂模式的组合常用于管理全局、可配置、需统一初始化的核心对象(如数据库连接池、日志实例、配置管理器等)。Go 本身没有类和构造函数,但可通过包级变量 + 惰性初始化 + 接口抽象来优雅实现:单例保证全局唯一,工厂负责按需创建具体类型,同时隐藏初始化细节。
用 sync.Once 实现线程安全的单例初始化
Go 标准库的 sync.Once 是实现单例最推荐的方式——它确保初始化函数仅执行一次,且并发安全,无需手动加锁。
示例:全局日志实例单例
var (
logger *zap.Logger
once sync.Once
)
func GetLogger() *zap.Logger {
once.Do(func() {
l, _ := zap.NewProduction()
logger = l
})
return logger
}
调用 GetLogger() 多次始终返回同一个实例,首次调用时完成初始化。
用接口+工厂函数解耦具体实现
定义统一接口,让不同环境或配置可返回不同实现(如开发用 console logger,生产用 file logger),再通过工厂函数封装创建逻辑:
type Logger interface {
Info(string, ...zap.Field)
Error(string, ...zap.Field)
}
func NewLogger(env string) Logger {
switch env {
case "dev":
l, _ := zap.NewDevelopment()
return l
case "prod":
l, _ := zap.NewProduction()
return l
default:
return zap.NewNop() // 空实现,避免 panic
}
}
此时单例可基于工厂结果构建:
var (
globalLogger Logger
loggerOnce sync.Once
)
func GetGlobalLogger(env string) Logger {
loggerOnce.Do(func() {
globalLogger = NewLogger(env)
})
return globalLogger
}
支持运行时配置与延迟初始化
实际项目中,配置往往来自命令行、环境变量或配置文件,不能在包初始化阶段硬编码。建议将配置参数传入工厂,并缓存配置+实例绑定关系:
- 使用结构体封装工厂状态,便于扩展(如支持多组 DB 实例)
- 用
map[string]instance缓存已创建的实例,键为配置标识(如"mysql-primary") - 结合
sync.RWMutex支持高频读、低频写场景
示例简版(无锁优化,适合简单场景):
type DBFactory struct {
instances map[string]*sql.DB
mu sync.RWMutex
}
func (f *DBFactory) GetDB(name string, dsn string) (*sql.DB, error) {
f.mu.RLock()
if db, ok := f.instances[name]; ok {
f.mu.RUnlock()
return db, nil
}
f.mu.RUnlock()
f.mu.Lock()
defer f.mu.Unlock()
if db, ok := f.instances[name]; ok {
return db, nil
}
db, err := sql.Open("mysql", dsn)
if err != nil {
return nil, err
}
f.instances[name] = db
return db, nil
}
避免常见陷阱
Go 中实现单例+工厂易踩的坑:
- 不要在 init() 中直接初始化全局对象:依赖未就绪(如 flag 未解析、env 未加载),导致配置错误或 panic
- 不要导出内部实例变量(如
var Logger *zap.Logger):破坏封装,外部可随意修改,应只暴露获取函数 - 慎用全局变量存储可变状态:单例不等于“全局可变容器”,状态变更应走明确方法(如
logger.SetLevel()),而非直接赋值 - 注意资源释放:单例若持有连接、文件句柄等,需提供
Close()或Shutdown()方法,并由主程序统一调用
今天带大家了解了的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
-
505 收藏
-
503 收藏
-
502 收藏
-
502 收藏
-
502 收藏
-
176 收藏
-
137 收藏
-
499 收藏
-
416 收藏
-
222 收藏
-
441 收藏
-
285 收藏
-
347 收藏
-
237 收藏
-
455 收藏
-
369 收藏
-
307 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习