单例设计模式
来源:dev.to
时间:2024-07-09 19:36:52 480浏览 收藏
今日不肯埋头,明日何以抬头!每日一句努力自己的话哈哈~哈喽,今天我将给大家带来一篇《单例设计模式》,主要内容是讲解等等,感兴趣的朋友可以收藏或者有更好的建议在评论提出,我都会认真看的!大家一起进步,一起学习!
单例设计模式是软件编程中最重要和最常用的设计模式之一。它确保类在应用程序运行时只有一个实例,并提供对该实例的全局访问点。在这篇文章中,我们将讨论 Singleton 的重要性,如何在 Golang 中实现它,以及它带来的好处,特别是在并发环境中。
什么是单例?
单例是一种将类的实例限制为单个实例的设计模式。它在需要单点控制或单个共享资源的情况下特别有用,例如:
- 配置管理器,需要集中应用程序设置。
- 数据库连接池,必须有效管理有限数量的连接。
- 记录器,其中日志一致性至关重要。
为什么使用单例?
我将列出一些关于 Pattern 实现的更有意义的观点,同时也表明并非一切都是美好的,我们可能会遇到一些问题。
好处
- 全局一致性:确保应用程序的所有点都使用相同的实例,提供数据和行为的一致性。
- 访问控制:集中控制实例的创建和访问,方便对象生命周期的维护和管理。
- 资源效率:避免不必要的创建多个实例,节省内存和处理资源。
缺点
- 测试难度:单例会使编写单元测试变得更加困难,因为它们引入了需要管理的全局状态。
- 增加耦合:过度使用单例会导致组件之间的耦合更紧密,从而使应用程序难以维护和发展。
实现单例
为了实现单例,我将使用 Golang。在这种语言中,我们必须特别注意并发性,以确保只创建一个实例,即使多个 goroutine 尝试同时访问该实例也是如此。
为了使我们的示例更接近现实世界,让我们为我们的应用程序创建一个记录器。记录器是应用程序中的常用工具,需要唯一以确保日志一致性。
1 - 定义结构
首先,我们定义我们想要拥有单个实例的结构。
包记录器 进口 ( “FMMT” “同步” ) 类型记录器结构{} var loggerInstance *Logger
2 - 实现NewInstance函数
NewInstance函数负责返回Singleton结构的单个实例。我们使用互斥锁来确保并发环境中的安全性,实现双重检查锁定以提高效率。
包记录器 进口 ( “FMMT” “同步” ) 类型记录器结构{} var 记录器*记录器 var mtx = &sync.Mutex{} func NewInstance() *记录器 { 如果记录器 == nil { mtx.Lock() 延迟 mtx.Unlock() 如果记录器 == nil { fmt.Println("创建新记录器") 记录器=&记录器{} } } 别的 { fmt.Println("记录器已创建") } 返回记录器 }
3 - 实现日志类型
日志工具总是有一些日志类型,比如Info只显示信息,Error显示错误等等。这也是过滤我们想要在应用程序中显示的信息类型的一种方法。
所以让我们创建一个方法来显示 Info 类型的日志。为此,我们将创建一个函数来接收日志消息并将其格式化为 INFO 格式。
包记录器 进口 ( “FMMT” “同步” “团队” ) 常量( 信息字符串 =“信息” ) 类型记录器结构{} var 记录器*记录器 var mtx = &sync.Mutex{} func NewInstance() *记录器 { 如果记录器 == nil { mtx.Lock() 延迟 mtx.Unlock() 如果记录器 == nil { fmt.Println("创建新记录器") 记录器=&记录器{} } } 别的 { fmt.Println("记录器已创建") } 返回记录器 } func (l *Logger) Info(消息字符串) { fmt.Printf("%s - %s: %sn", time.Now().UTC().Format(time.RFC3339Nano), INFO, 消息) }
4 - 使用记录器
为了使用我们的新记录器,我们将在主包中实例化它并创建一个日志来查看此实现是如何工作的。
包主 进口 ( “playground-go/pkg/logger” ) 函数主() { 日志 := logger.NewInstance() log.Info("这是日志的示例") }
这是我们运行程序时的结果:
创建新记录器 2024-07-03T19:34:57.609599Z - 信息:这是日志的示例
如果我们想测试NewInstance是否真的保证只有一个实例在运行,我们可以做以下测试。
包主 进口 ( “FMMT” “playground-go/pkg/logger” ) 函数主() { 日志 := logger.NewInstance() log.Info("这是日志的示例") log2 := logger.NewInstance() log2.Info("这是日志的另一个例子") 如果日志 == log2 { fmt.Println("同一实例") } 别的 { fmt.Println("不同的实例") } }
我们的日志已更改,现在我们可以看到我们阻止了新实例的创建:
创建新记录器 2024-07-03T19:45:19.603783Z - 信息:这是日志的示例 记录器已创建 2024-07-03T19:45:19.603793Z - 信息:这是日志的另一个示例 同一实例
结论
单例模式是一种强大的工具,可确保应用程序运行时仅存在特定类的一个实例。在记录器示例中,我们了解了如何应用此模式来确保整个应用程序中的日志一致性。
希望这可以帮助您更好地理解 Golang 中的 Singleton。
今天关于《单例设计模式》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
-
505 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
159 收藏
-
144 收藏
-
108 收藏
-
332 收藏
-
251 收藏
-
223 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习