[go] 单例模式

单例模式

确保类只有一个实例,并提供一个全局的访问点。

  • 单例(Singleton)类声明了一个名为getInstance的方法来返回其所属类的一个相同实例。

  • 单例的构造函数必须对客户端的代码隐藏。调用getInstance方法必须是获取单例对象的唯一方式。

优缺点

1.优点

你可以肯定,一个类只有一个实例。

有助于获得该实例的全局访问点。

仅在首次请求单例对象时对其进行初始化。

2.缺点

  • 它违反了单一责任原则。该模式同时解决了两个问题。
  1. 保证一个类只有一个实例
  2. 为该实例提供一个全局访问节点
  • 单例模式可以掩盖糟糕的设计。例如,当程序组件对彼此了解太多。

  • 不是线程安全的,你写的代码要确保为所有线程只创建一个实例。

  • 单例的客户端代码单元测试可能会比较困难, 因为许多测试框架以基于继承的方式创建模拟对象。 由于单例类的构造函数是私有的, 而且绝大部分语言无法重写静态方法, 所以你需要想出仔细考虑模拟单例的方法。 要么干脆不编写测试代码, 或者不使用单例模式。

使用场景

  • 如果程序中的某个类对于所有客户端只有一个可用的实例, 可以使用单例模式。
  • 如果你需要更加严格地控制全局变量, 可以使用单例模式。

参考代码

这里我们分两个例子吧,一个是简单的单例实现,另外一个是解决异步情况可能会出现的问题
sample

go 复制代码
type single struct{}

var normalSingleInstance *single

func getNormalSingle() *single {
	if normalSingleInstance == nil {
		fmt.Println("creating single instance")
		normalSingleInstance = new(single)
	} else {
		fmt.Println("single instance already created")
	}
	return normalSingleInstance
}

func main() {
	for i := 0; i < 10; i++ {
		getNormalSingle()
	}
}

输出:

go 复制代码
creating single instance
single instance already created
single instance already created
single instance already created
single instance already created
single instance already created
single instance already created
single instance already created
single instance already created
single instance already created

multithread

go 复制代码
type single struct{}

var lock = &sync.Mutex{}

var multiThreadSafeInstance *single

func getMultiThreadSafeInstance() *single {
	if multiThreadSafeInstance == nil {
		lock.Lock()
		defer lock.Unlock()
		if multiThreadSafeInstance == nil {
			fmt.Println("creating multi-thread safe instance now.")
			multiThreadSafeInstance = new(single)
		} else {
			fmt.Println("multi-thread safe instance already created.")
		}
	} else {
		fmt.Println("multi-thread safe instance already created.")
	}

	return multiThreadSafeInstance
}

func main() {
	for i := 0; i < 2; i++ {
		go func() {
			for j := 0; j < 5; j++ {
				getMultiThreadSafeInstance()
			}
		}()
	}
	select {}
}

输出:

go 复制代码
creating multi-thread safe instance now. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created. 
multi-thread safe instance already created.
相关推荐
白兰地空瓶1 分钟前
当神经网络跑在浏览器里:brain.js 前端机器学习实战
javascript·人工智能
我爱画页面2 分钟前
vue3封装table组件及属性介绍
开发语言·javascript·ecmascript
逻极3 分钟前
Next.js vs Vue.js:2025年全栈战场,谁主沉浮?
开发语言·javascript·vue.js·reactjs
一枚前端小能手3 分钟前
🗂️ Blob对象深度解析 - 从文件处理到内存优化的完整实战指南
前端·javascript
杰克尼4 分钟前
vue-day02
前端·javascript·vue.js
一只小阿乐4 分钟前
vue3 中实现父子组件v-model双向绑定 总结
前端·javascript·vue.js·vue3·组件·v-model语法糖
qq_3380329211 分钟前
Vue 3 的<script setup> 和 Vue 2 的 Options API的关系
前端·javascript·vue.js
lumi.12 分钟前
Vue Router页面跳转指南:告别a标签,拥抱组件化无刷新跳转
前端·javascript·vue.js
Mintopia16 分钟前
🧠 一文吃透 Next.js 中的 JWT vs Session:底层原理+幽默拆解指南
前端·javascript·全栈
前端开发爱好者19 分钟前
字节出手!「Vue Native」真的要来了!
前端·javascript·vue.js