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

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


文章目录

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

单例模式

1. 介绍

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

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

2. 应用场景

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

3. 实现

3.1 饿汉式
go 复制代码
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 懒汉模式

线程不安全

go 复制代码
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,然后返回这个实例

线程安全

go 复制代码
type Singleton struct{}

var singleInstance *Singleton

// 懒汉式(线程安全)
var mutex sync.Mutex

func 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实例,从而保证了单例模式的正确性。

go 复制代码
type Singleton struct{}

var singleInstance *Singleton

var once sync.Once

func 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 方法会确保传入的匿名函数只会被执行一次

源码

go 复制代码
package singleton

import (
	"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.Mutex

func GetInstance3() *Singleton {
	fmt.Println("GetInstance3")
	mutex.Lock()
	defer mutex.Unlock()
	if singleInstance == nil {
		singleInstance = &Singleton{}
	}
	return singleInstance
}

// 使用sync.Once实现
var once sync.Once

func 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)
}
相关推荐
阿超爱嵌入式10 分钟前
STM32学习笔记之RCC模块(实操篇)
笔记·stm32·学习
yanyu-yaya10 分钟前
devextreme-react/scheduler 简单学习
前端·学习·react.js
淬渊阁17 分钟前
汇编学习之《运算和逻辑指令》
汇编·学习
别来无恙20218 分钟前
算法设计学习4
c++·学习
小王努力学编程27 分钟前
动态规划学习——回文子串系列问题【C++】
c++·学习·算法·leetcode·动态规划
Cynthia的梦1 小时前
Linux学习-Linux进程间通信(IPC)聊天程序实践指南
linux·运维·学习
卡戎-caryon1 小时前
【Linux网络与网络编程】03.UDP Socket编程
linux·服务器·网络·笔记·单例模式·udp·网络通信
nuise_1 小时前
李宏毅机器学习笔记06 | 鱼和熊掌可以兼得的机器学习 - 内容接宝可梦
人工智能·笔记·机器学习
代码AC不AC1 小时前
【数据结构】队列
c语言·数据结构·学习·队列·深度讲解
生信小鹏2 小时前
Nature旗下 | npj Digital Medicine | 图像+转录组+临床变量三合一,多模态AI预测化疗反应,值得复现学习的完整框架
人工智能·学习·免疫治疗·scrna-seq·scrna