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)
}
相关推荐
StickToForever3 小时前
第4章 信息系统架构(五)
经验分享·笔记·学习·职场和发展
闲猫5 小时前
go orm GORM
开发语言·后端·golang
丁卯4045 小时前
Go语言中使用viper绑定结构体和yaml文件信息时,标签的使用
服务器·后端·golang
leegong231116 小时前
学习PostgreSQL专家认证
数据库·学习·postgresql
敲敲敲-敲代码6 小时前
【SQL实验】触发器
数据库·笔记·sql
Moonnnn.7 小时前
51单片机学习——动态数码管显示
笔记·嵌入式硬件·学习·51单片机
南宫生8 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
技术小齐8 小时前
网络运维学习笔记 016网工初级(HCIA-Datacom与CCNA-EI)PPP点对点协议和PPPoE以太网上的点对点协议(此处只讲华为)
运维·网络·学习
竹言笙熙8 小时前
代码审计初探
学习·web安全
日记成书8 小时前
物联网智能项目
物联网·学习