当前位置: 首页 > news >正文

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

前言

单例模式是最简单的一种模式。在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

相关文章:

  • 物联网控制器:一台顶N台!路由器、PLC控制器、网关、工控机……
  • PowerShell批量处理文件名称/内容的修改
  • 【二进制安全作业】250617课上作业4 - start
  • Ntfs!LfsGetLbcb函数分析之Ntfs!LfsPrepareLfcbForLogRecord
  • Sentinel 服务限流机制
  • opencv依据图像类型读取图像像素点
  • 软件公司进军无人机领域的战略指南与生态合作全景-优雅草卓伊凡
  • Tomcat项目本地部署(Servlet为例)
  • 性能测试-jmeter实战3
  • jmeter工具简单认识
  • Vue + AbortController 请求取消弹窗 hook 封装
  • 【Qt】QTableWidget表格右键菜单
  • 键盘觉醒:Raycast 把 Mac 变成「AI 指令战舰」
  • joomla 使用nginx服务器只能打开首页,其他页面404的解决方案
  • 16、Redis底层数据结构
  • 样式失控:Word文档格式混乱的根源与对策
  • 【Flink实战】 Flink SQL 中处理字符串 `‘NULL‘` 并转换为 `BIGINT`
  • 【Ambari3.0.0 部署】Step4—时间同步(Chrony)-适用于el8
  • Xilinx XC7A12T‑1CPG238I Artix‑7 FPGA
  • Java 编程之代理模式
  • 青岛网站建设设计/最近的新闻大事10条
  • ps做网站图/爱站工具下载
  • 资讯网站 怎么做/房地产十大营销手段
  • 网站有后台更新不了/百度云资源链接分享群组
  • 大连做公司网站/宁波seo推广优化公司
  • 网站制作如何做/品牌推广软文