0

0

如何在Golang中实现单例模式_Golang单例模式实现方法

P粉602998670

P粉602998670

发布时间:2026-01-23 15:49:02

|

244人浏览过

|

来源于php中文网

原创

Go单例靠包级变量+sync.Once实现,线程安全且延迟初始化;不用init因无法按需、不支持错误返回、难测试;禁用if-nil手动实现以防竞态。

如何在golang中实现单例模式_golang单例模式实现方法

Go 语言里没有“类”和“构造函数”,所以单例不是靠私有化构造器实现的,而是靠包级变量 + sync.Once 控制初始化时机 —— 这是最安全、最常用的方式。

sync.Once 保证全局唯一实例

直接声明一个包级指针变量,配合 sync.OnceDo 方法确保 init 函数只执行一次。这是 Go 官方推荐的单例写法,线程安全且无竞态风险。

  • sync.Once 内部使用原子操作和互斥锁,比手写 if instance == nil + mutex.Lock() 更可靠
  • 初始化逻辑放在闭包或独立函数里,避免在变量声明时就执行(防止 init 循环或依赖未就绪)
  • 返回指针类型,方便后续方法定义为值接收者或指针接收者
package singleton

import "sync"

type Config struct {
    Timeout int
    Env     string
}

var (
    instance *Config
    once     sync.Once
)

func GetConfig() *Config {
    once.Do(func() {
        instance = &Config{
            Timeout: 30,
            Env:     "prod",
        }
    })
    return instance
}

为什么不用 init() 函数?

init() 确实只执行一次,但它在包加载时就运行,无法按需延迟初始化,也不支持带参数或错误返回 —— 实际项目中单例常需读配置、连数据库、校验权限,这些都可能失败。

  • init() 不能返回 error,出错只能 panic,不可控
  • 无法在测试中重置或替换单例(比如 mock 数据库连接)
  • 如果单例依赖其他尚未 init 的包,会触发隐式依赖顺序问题

带错误返回的单例(如初始化 DB 连接)

当初始化可能失败(比如打开文件、连接 Redis),需要把 error 暴露给调用方,并缓存失败状态避免重复尝试。

施乐在线订单系统
施乐在线订单系统

一套简单的数据库结构的在线订单系统,采用数据库存储格式,方便的实现产品的在线订购,带有后台管理模块用户名为: admin 密码: ojdj22 修改密码方法如下, 更改 ckpwd.asp 中 第三行 if (user="用户名" and pwd="密码") 即可

下载

立即学习go语言免费学习笔记(深入)”;

  • 用两个包级变量:一个存实例,一个存 error
  • 首次调用时初始化,之后直接返回缓存的结果(无论成功或失败)
  • 不建议在 GetXXX() 中 panic,应由上层决定如何处理 error
package db

import (
    "database/sql"
    "sync"
)

var (
    instance *sql.DB
    err      error
    once     sync.Once
)

func GetDB(dsn string) (*sql.DB, error) {
    once.Do(func() {
        instance, err = sql.Open("mysql", dsn)
        if err == nil {
            err = instance.Ping()
        }
    })
    return instance, err
}

注意:不要用全局变量 + if 判断模拟单例

这种写法看似简洁,但存在竞态风险,尤其在高并发场景下可能创建多个实例:

// ❌ 危险!可能创建多个实例
var instance *Config

func GetConfig() *Config {
    if instance == nil { // 多个 goroutine 同时通过判断
        instance = &Config{} // 多次赋值
    }
    return instance
}

即使加了 mutex,也容易漏锁或死锁;而 sync.Once标准库专为此设计的原语,无需自己造轮子。真正要注意的是:别在单例方法里做耗时操作(比如每次调用都查一次 etcd),单例只管“实例创建”,不负责“每次调用逻辑”。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

191

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

273

2025.06.17

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

10

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.9万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 805人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号