GO设计模式——4、单例模式(创建型)

目录

[单例模式(Singleton Pattern)](#单例模式(Singleton Pattern))

优缺点

使用场景

饿汉式和懒汉式单例模式


单例模式(Singleton Pattern)

单例模式(Singleton Pattern)是一个类只允许创建一个对象(或者实例),那这个类就是一个单例类,这种设计模式就叫作单例设计模式,简称单例模式。单例模式的要点有三个:一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。单例模式通常适用于需要共享资源、管理全局状态或控制某个唯一实例的情况。在开发中,单例模式可以帮助我们解决数据结构与操作之间的耦合问题,实现代码的复用和扩展。

单例模式的核心思想是:通过将类的构造函数私有化,防止外部代码直接实例化该类,然后提供一个静态方法或全局变量来获取类的唯一实例。具体而言,单例模式包含以下几个关键要素:(1)私有的构造函数(Private Constructor):用于限制类的实例化,使其只能在类的内部进行。(2)静态变量或静态方法(Static Variable/Method):用于保存类的唯一实例或提供全局访问点。通过将构造函数私有化,外部代码无法直接实例化类,只能通过静态方法或全局变量来获取类的唯一实例。当第一次获取实例时,单例模式会创建一个实例,并将其保存起来,以供后续调用使用。

优缺点

(1)优点:

  • 在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例(比如管理学院首页页面缓存)。
  • 避免对资源的多重占用(比如写文件操作)。

(2)缺点:没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。

使用场景

  • 要求生产唯一序列号。
  • WEB 中的计数器,不用每次刷新都在数据库里加一次,用单例先缓存起来。
  • 创建的一个对象需要消耗的资源过多,比如 I/O 与数据库的连接等。

饿汉式和懒汉式单例模式

两种常见实现方式:饿汉式单例模式和懒汉式单例模式。饿汉式单例模式适用于在程序初始化时就需要创建实例,并且该实例在整个程序生命周期内都需要被使用的情况。例如,全局的配置管理器或系统的日志记录器可以使用饿汉式单例模式来确保全局唯一性和一致性。懒汉式单例模式适用于需要延迟实例化的情况,即只有在需要时才创建实例。例如,某些资源消耗较大的对象,或者需要根据运行时的条件来确定具体实例的情况下,可以使用懒汉式单例模式。

1. 饿汉式单例模式

饿汉式单例模式是一种在类加载时就创建唯一实例的单例模式。在这种模式下,实例在整个程序的生命周期中始终存在,无论是否被使用。在GetConfigManager()方法中返回配置管理器的唯一实例。配置管理器包含一个configData字典,用于保存配置数据。在程序初始化时创建了配置管理器的实例,并通过instance变量保存起来。

饿汉式单例模式代码实现:

Go 复制代码
package main

import "fmt"

type ConfigManager struct {
    configData map[string]string
}

var instance = &ConfigManager{
    configData: make(map[string]string),
}

func GetConfigManager() *ConfigManager {
    return instance
}

func (cm *ConfigManager) Set(key, value string) {
    cm.configData[key] = value
}

func (cm *ConfigManager) Get(key string) string {
    return cm.configData[key]
}

func main() {
    configManager := GetConfigManager()
    configManager.Set("key1", "value1")
    fmt.Println("key1:", configManager.Get("key1"))

    configManager1 := GetConfigManager()
    configManager1.Set("key1", "value2")
    fmt.Println("key1:", configManager.Get("key1"))
}

2. 懒汉式单例模式

懒汉式单例模式是一种在需要时才创建实例的单例模式。在这种模式下,实例的创建延迟到第一次被使用时才进行。使用了sync.Once来确保GetLogger()方法只被执行一次,从而保证只有一个实例被创建。日志记录器包含了一个日志列表,可以通过AddLog()方法添加日志,以及通过PrintLogs()方法打印所有日志。

懒汉式单例模式代码实现:

Go 复制代码
package main

import (
    "fmt"
    "sync"
)

type Logger struct {
    logs []string
}

var instancel *Logger
var once sync.Once

func GetLogger() *Logger {
    once.Do(func() {
       instancel = &Logger{
          logs: []string{},
       }
    })
    return instancel
}
func (l *Logger) AddLog(log string) {
    l.logs = append(l.logs, log)
}
func (l *Logger) printLogs() {
    for _, log := range l.logs {
       fmt.Println(log)
    }
}

func main() {
    logger := GetLogger()
    logger.AddLog("Log 1")
    logger.AddLog("log2")

    logger1 := GetLogger()
    logger1.AddLog("log3")

    logger.printLogs()

    logger1.printLogs()
}
相关推荐
wrx繁星点点27 分钟前
状态模式(State Pattern)详解
java·开发语言·ui·设计模式·状态模式
明月看潮生1 小时前
青少年编程与数学 02-003 Go语言网络编程 09课题、Cookie
青少年编程·golang·网络编程·编程与数学
明月看潮生2 小时前
青少年编程与数学 02-003 Go语言网络编程 15课题、Go语言URL编程
开发语言·网络·青少年编程·golang·编程与数学
明月看潮生2 小时前
青少年编程与数学 02-003 Go语言网络编程 14课题、Go语言Udp编程
青少年编程·golang·网络编程·编程与数学
金池尽干2 小时前
设计模式之——观察者模式
观察者模式·设计模式
也无晴也无风雨2 小时前
代码中的设计模式-策略模式
设计模式·bash·策略模式
hlsd#3 小时前
go 集成go-redis 缓存操作
redis·缓存·golang
捕鲸叉11 小时前
MVC(Model-View-Controller)模式概述
开发语言·c++·设计模式
wrx繁星点点12 小时前
享元模式:高效管理共享对象的设计模式
java·开发语言·spring·设计模式·maven·intellij-idea·享元模式
凉辰12 小时前
设计模式 策略模式 场景Vue (技术提升)
vue.js·设计模式·策略模式