Golang学习笔记_27——单例模式

ops/2025/1/21 13:40:01/

Golang学习笔记_24——泛型
Golang学习笔记_25——协程Golang学习笔记_25——协程
Golang学习笔记_26——通道


文章目录

    • 单例模式
      • 1. 介绍
      • 2. 应用场景
      • 3. 实现
        • 3.1 饿汉式
        • 3.2 懒汉模式
    • 源码


单例模式

1. 介绍

单例模式是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点来访问这个实例。

这种模式在很多场景下非常有用,比如数据库连接池、日志系统等,这些场景中我们通常希望在整个应用程序中只有一个对象来负责相关的操作,避免资源的浪费和数据的不一致

2. 应用场景

  • 数据库连接池:在应用程序中,数据库连接的创建和销毁是比较消耗资源的操作。使用单例模式可以确保整个应用程序只有一个数据库连接池实例,多个地方需要获取数据库连接时都从这个连接池中获取,这样可以有效地管理数据库连接,提高性能并节省资源。
  • 日志系统:一个应用程序通常只需要一个日志记录器来统一记录各种操作信息。单例模式可以保证整个系统只有一个日志记录器实例,所有的日志记录操作都通过这个实例来完成,方便对日志进行统一管理和配置。
  • 配置管理:对于应用程序的配置信息,如服务器端口号、数据库连接参数等,使用单例模式可以确保整个应用程序只有一个配置管理实例,这样可以方便地在不同的模块中获取和修改配置信息,并且保证配置信息的一致性。

3. 实现

3.1 饿汉式
type Singleton struct{}var singleInstance *Singleton = &Singleton{}func GetInstance() *Singleton {return singleInstance
}func test1() {instance1 := GetInstance()instance2 := GetInstance()fmt.Println(instance1 == instance2)
}

说明
这种方式在程序启动时就初始化了单例实例singleInstance。&Singleton{}创建了一个Singleton结构体的实例,并将其赋值给singleInstance。

GetInstance函数只是简单地返回这个已经初始化好的实例。这种方式被称为饿汉式,因为实例是在程序开始时就 “急切” 地创建好了,而不管是否马上会被用到。

优点是实现简单,并且在多线程环境下也是安全的,因为实例在任何线程访问之前就已经创建好了。

缺点是如果单例的初始化过程很复杂或者资源消耗大,可能会导致程序启动变慢。

3.2 懒汉模式

线程不安全


type Singleton struct{}var singleInstance *Singleton
// 懒汉式(非线程安全)
func GetInstance2() *Singleton {fmt.Println("GetInstance2")if singleInstance == nil {singleInstance = &Singleton{}}return singleInstance
}func test1() {
instance1 := GetInstance()
instance2 := GetInstance()
fmt.Println(instance1 == instance2)
}

懒汉式单例模式。在GetInstance函数中,首先检查singleInstance是否为nil。如果是,就创建一个Singleton结构体的新实例并赋值给singleInstance,然后返回这个实例

线程安全

type Singleton struct{}var singleInstance *Singleton// 懒汉式(线程安全)
var mutex sync.Mutexfunc GetInstance3() *Singleton {fmt.Println("GetInstance3")mutex.Lock()defer mutex.Unlock()if singleInstance == nil {singleInstance = &Singleton{}}return singleInstance
}func test1() {
instance1 := GetInstance()
instance2 := GetInstance()
fmt.Println(instance1 == instance2)
}

为了在多线程环境下正确地实现懒汉式单例模式,引入了互斥锁sync.Mutex

在GetInstance函数中,首先调用mutex.Lock()获取锁,这确保了同一时刻只有一个线程能够进入临界区

使用defer mutex.Unlock()可以保证在函数返回之前释放锁。这样,即使多个线程同时调用GetInstance函数,也能保证只有一个线程会创建singleInstance实例,从而保证了单例模式的正确性。

type Singleton struct{}var singleInstance *Singletonvar once sync.Oncefunc GetInstance4() *Singleton {fmt.Println("GetInstance4")once.Do(func() {fmt.Println("just once!")singleInstance = &Singleton{}})return singleInstance
}func test1() {
instance1 := GetInstance()
instance2 := GetInstance()
fmt.Println(instance1 == instance2)
}

once.Do 方法会确保传入的匿名函数只会被执行一次

源码

package singletonimport ("fmt""sync"
)type Singleton struct{}// 饿汉式
// var singleInstance *Singleton = &Singleton{} // 懒汉式直接在程序运行时创建
func GetInstance1() *Singleton {fmt.Println("GetInstance1")return singleInstance
}var singleInstance *Singleton// 懒汉式(非线程安全)
func GetInstance2() *Singleton {fmt.Println("GetInstance2")if singleInstance == nil {singleInstance = &Singleton{}}return singleInstance
}// 懒汉式(线程安全)
var mutex sync.Mutexfunc GetInstance3() *Singleton {fmt.Println("GetInstance3")mutex.Lock()defer mutex.Unlock()if singleInstance == nil {singleInstance = &Singleton{}}return singleInstance
}// 使用sync.Once实现
var once sync.Oncefunc GetInstance4() *Singleton {fmt.Println("GetInstance4")once.Do(func() {fmt.Println("just once!")singleInstance = &Singleton{}})return singleInstance
}// 测试方法
func test1() {//instance1 := GetInstance1()//instance2 := GetInstance1()//instance1 := GetInstance2()//instance2 := GetInstance2()//instance1 := GetInstance3()//instance2 := GetInstance3()instance1 := GetInstance4()instance2 := GetInstance4()fmt.Println(instance1 == instance2)
}

http://www.ppmy.cn/ops/151921.html

相关文章

数据结构——链表和单向链表

1、链表的介绍 (1)定义 链表是一种链式存储的线性表 链表是一种基本的数据结构,它由一系列节点组成,每个节点包含一个值和指向下一个节点的指针 节点如下图所示: 与数组不同,链表中的节点不一定是连续的…

【漫话机器学习系列】054.极值(Extrema)

极值(Extrema) 定义 极值是数学分析和优化问题中的一个核心概念,指函数在某个定义域内取得的最大值或最小值。根据极值的性质,可以将其分为两类: 局部极值(Local Extrema):函数在…

Micrometer+Zipkin 分布式链路追踪

MicrometerZipkin 分布式链路追踪(Distributed Tracing)是一种用于监控和分析分布式系统性能的技术。它允许开发人员和运维人员追踪请求在分布式系统中的传播路径,包括跨服务调用、数据库访问、缓存查询等操作。通过分布式链路追踪&#xff0…

WPF基础 | 初探 WPF:理解其核心架构与开发环境搭建

WPF基础 | 初探 WPF:理解其核心架构与开发环境搭建 一、前言二、WPF 核心架构2.1 核心组件2.2 布局系统2.3 数据绑定机制2.4 事件处理机制 三、WPF 开发环境搭建3.1 安装 Visual Studio3.2 创建第一个 WPF 应用程序 结束语优质源码分享 WPF基础 | 初探 WPF&#xff…

游戏引擎学习第79天

当前任务回顾 我们目前的工作重点是碰撞检测的更新,特别是将游戏的世界表示方式扩展到三维空间。尽管游戏本身是二维的,但我们希望它能够在三维空间中处理更多的内容,以支持那些需要考虑高度的游戏元素,如楼层、台阶等。我们的目…

【HF设计模式】06-命令模式

声明:仅为个人学习总结,还请批判性查看,如有不同观点,欢迎交流。 摘要 《Head First设计模式》第6章笔记:结合示例应用和代码,介绍命令模式,包括遇到的问题、采用的解决方案、遵循的 OO 原则、…

WinHttp API接口辅助类实现GET POST网络通讯

1、简述 近期需要在MFC基础上开发网络Http通讯,开始使用的WinINet进行通讯,后面发现WinINet对连接超时这块不支持设置,在网上搜索了几种方式效果都不太好,于是决定用WinHttp API接口进行通讯,分别对GET、POST进行了封装。 2、使用到接口 2.1、WinHttpOpen WinHttpOpen 是…

Golang Gin系列-2:搭建Gin 框架环境

开始网络开发之旅通常是从选择合适的工具开始的。在这个全面的指南中,我们将引导你完成安装Go编程语言和Gin框架的过程,Gin框架是Go的轻量级和灵活的web框架。从设置Go工作空间到将Gin整合到项目中,本指南是高效而强大的web开发路线图。 安装…