首页 > 其他分享 >[设计模式]创建型模式-单例模式

[设计模式]创建型模式-单例模式

时间:2024-02-19 11:45:50浏览次数:23  
标签:__ self 模式 instance 单例 ._ 设计模式 cls

前言

单例模式是最简单的一种模式。在Go中,单例模式指的是全局只有一个实例,并且它负责创建自己的对象。单例模式有减少内存和系统资源开销、防止多个实例产生冲突等优点。

因为单例模式保证了实例的全局唯一性,并且只被初始化一次,所以比较适合全局共享一个实例,且只需要被初始化一次的场景,例如数据库实例、全局配置、全局任务池等。

单例模式又分为饿汉方式和懒汉方式。饿汉方式是指全局的单例实例在包被加载时创建,而懒汉方式指全局的单例实例在第一次被使用时创建。其中懒汉方式是开源项目中使用最多的方式。

示例代码

Go

懒汉方式的缺点是非并发安全,实际使用中一般加锁,或者使用sync.Once

package singleton  
  
import "sync"  
  
type Singleton interface {  
    foo()  
}  
  
type singleton struct{}  
  
func (s singleton) foo() {}  
  
var (  
    instance *singleton  
    once     sync.Once  
)  
  
func GetInstance() Singleton {  
    once.Do(func() {  
       instance = &singleton{}  
    })  
  
    return instance  
}

单元测试

package singleton  
  
import (  
    "sync"  
    "testing")  
  
const parCount = 100  
  
func TestSingleton(t *testing.T) {  
    ins1 := GetInstance()  
    ins2 := GetInstance()  
    if ins1 != ins2 {  
       t.Fatal("instance is not equal")  
    }  
}  
  
func TestParallelSingleton(t *testing.T) {  
    start := make(chan struct{})  
    wg := sync.WaitGroup{}  
    wg.Add(parCount)  
  
    instance := [parCount]Singleton{}  
    for i := 0; i < parCount; i++ {  
       go func(index int) {  
          <-start  
          instance[index] = GetInstance()  
          wg.Done()  
       }(i)  
    }  
    close(start)  
    wg.Wait()  
    for i := 1; i < parCount; i++ {  
       if instance[i] != instance[i-1] {  
          t.Fatal("instance is not equal")  
       }  
    }  
}

Python

python的包是天然的单例模式,只要放到单独的包中,import时就是引用的单例。

如果要在一个包内使用设计模式,也有以下几种方式。

使用函数装饰器实现单例

def singleton(cls):  
    _instance = {}  
  
    def inner():  
        if cls not in _instance:  
            _instance[cls] = cls()  
        return _instance[cls]  
  
    return inner  
  
  
@singleton  
class MyCls:  
    def __init__(self):  
        pass  
  
  
if __name__ == "__main__":  
    a = MyCls()  
    b = MyCls()  
    print(id(a) == id(b))  # 输出结果应为 True

使用类装饰器实现单例

class Singleton:  
    def __init__(self, cls):  
        self._cls = cls  
        self._instance = {}  
  
    def __call__(self):  
        if self._cls not in self._instance:  
            self._instance[self._cls] = self._cls()  
        return self._instance[self._cls]  
  
  
@Singleton  
class MyCls:  
    def __init__(self):  
        pass  
  
  
if __name__ == "__main__":  
    a = MyCls()  
    b = MyCls()  
    print(id(a) == id(b))  # 输出结果应该是True

参考

标签:__,self,模式,instance,单例,._,设计模式,cls
From: https://www.cnblogs.com/XY-Heruo/p/18020746

相关文章

  • [设计模式]行为型模式-策略模式
    前言策略模式定义了一系列算法,并将每个算法封装起来,使它们可以互相替换,且算法的变换不会影响使用算法的客户。在项目开发中,我们经常要根据不同的场景,采取不同的措施,也就是不同的策略。假设我们需要对a、b这两个整数进行计算,根据条件的不同,需要执行不同的计算方式。我们可以把所......
  • Unity常用设计模式之代理模式
    代理模式是一种常用的设计模式,它允许一个对象代表另一个对象进行访问。在Unity中,代理模式经常被用来控制对某些对象的访问,以便在访问对象的同时进行一些额外的操作,比如权限控制、缓存、日志记录等。代理模式的结构包括三个角色:抽象主题(Subject)、真实主题(RealSubject)和代理(Proxy)。......
  • 软件架构模式之第五章:事件驱动架构
     第五章:事件驱动架构近年来,事件驱动架构风格显著增长并广泛应用,我们对其理解方式也发生了改变。这种高采用率并不令人意外,因为事件驱动架构能够解决复杂的非确定性工作流和高度反应和响应的系统等难题。此外,新技术、工具、框架和基于云的服务使得事件驱动架构比以往更易访问......
  • stm32芯片的SPI接口调试总结之轮询模式
    一概念1组成SPI系统可直接与各个厂家生产的多种标准外围器件接口,它只需4条线:串行时钟线(SCK)、主机输入/从机输出数据线(MISO)、主机输出/从机输入数据线(MOSI)和低电平有效的从机选择线(NSS)。(1)MISO:主设备输入/从设备输出引脚。该引脚在从模式下发送数据,在主模式下接收数......
  • 记一次centos7.9崩溃恢复操作(limits.conf配置失误),救援模式
    引起故障的原因:调整了操作系统的内核参数文件limits.conf,* softnproc131072* hardnproc131072* softnofile65536* hardnofile131072 以上的参数都扩大了10倍,ssh登录主机就开始异常了,连上去后就直接断开了,表象就是无法新建ssh会话连接,当前的用户有sudo免密操......
  • 策略模式实现短信平台切换
    如果你需要更通用的代码,以便在不同的短信平台之间进行切换,你可以使用策略模式来实现。策略模式允许你在运行时选择不同的策略(即短信平台),而无需改变主要的业务逻辑。首先,创建一个SmsPlatform接口,定义发送短信的方法:publicinterfaceSmsPlatform{voidsendSms(Stringphone......
  • 桥接模式
    桥接模式(Bridge模式)是一种结构型设计模式,它将抽象部分与它的实现部分分离,使它们都可以独立地变化。这种模式的核心意图是将抽象和实现解耦,使得抽象部分和实现部分可以独立地进行扩展和变化,而不会互相影响。桥接模式包含两个主要角色:抽象化(Abstraction)角色和实现化(Implementor)角色......
  • 【数据结构】串的表示与模式匹配算法
    串串是内容受限的线性表(栈和队列是操作受限的线性表)串(string)是零个或多个任意字符组成的有限序列S:串名a1a2a3...an:串值n:串长当n=0时,表示空串,空串用\(\phi\)表示子串:一个串中任意个连续字符组成的子序列(含空串)例如“abc”的子串有“”、“a”、“b”、"c"、"ab"......
  • 软件架构模式之第四章:微内核架构
     第四章:微内核架构微内核架构风格是一种高度灵活且可扩展的架构,允许开发人员或最终用户以插件形式轻松地为现有应用程序添加额外功能和特性,而不会对系统的核心功能产生任何影响。因此,微内核架构常被称为“插件化架构”(这也是其另一个常见名称)。该架构风格非常适合产品型应用......
  • 在k8S中,所支持的存储供应模式有哪些?
    在Kubernetes(k8S)中,支持多种存储供应模式以满足不同场景下的持久化存储需求。主要的存储供应模式包括:静态供应(ManualProvisioning)在这种模式下,集群管理员手动创建PersistentVolume(PV)资源,并配置其具体的存储类型、大小和访问模式。然后用户通过创建与之匹配的Persi......