一起学习 Go 语言设计模式之单例模式(上)

简介: 单例模式很容易记住。就像名称一样,它只能提供对象的单一实例,保证一个类只有一个实例,并提供一个全局访问该实例的方法。

单例模式的概念

单例模式很容易记住。就像名称一样,它只能提供对象的单一实例,保证一个类只有一个实例,并提供一个全局访问该实例的方法。


在第一次调用该实例时被创建,然后在应用程序中需要使用该特定行为的所有部分之间重复使用。

单例模式结构

image.png

单例模式的使用场景

你会在许多不同的情况下使用单例模式。比如:


  • 当你想使用同一个数据库连接来进行每次查询时
  • 当你打开一个安全 Shell(SSH)连接到一个服务器来做一些任务时。而不想为每个任务重新打开连接
  • 如果你需要限制对某些变量或空间的访问,你可以使用一个单例作为作为这个变量的门(在 Go 中使用通道可以很好地实现)
  • 如果你需要限制对某些空间的调用数量,你可以创建一个单例实例使得这种调用只在可接受的窗口中进行


单例模式还有跟多的用途,这里只是简单的举出一些。


先来看 Java  中的单例模式实现

public class Singleton {    private static Singleton uniqueInstance; // 一个静态变量持有 Singleton 类的唯一实例     private Singleton() {}  // 构造器声明为私有,只有 Singleton 可以实例化这个类     // getInstance()方法提供了一种实例化该类的方式,也返回它的一个实例   public static Singleton getInstance() {       if (uniqueInstance == null) { // 如果 uniqueInstance 为空,表示还没有创建实例...            // 通过构造器的私有方法实例化 Singleton,并赋值给 uniqueInstance            // 注意,如果我们不需要这个实例,它就不会被创建,这就是延迟实例化(lazy instantiation)            uniqueInstance = new Singleton();         }            // 如果 uniqueinstance 不为空,说明之前已经创建过对象,直接跳转到 return 语句      return uniqueInstance;    }}

复制代

public class Singleton {
    private static Singleton uniqueInstance; // 一个静态变量持有 Singleton 类的唯一实例
    private Singleton() {}  // 构造器声明为私有,只有 Singleton 可以实例化这个类
    // getInstance()方法提供了一种实例化该类的方式,也返回它的一个实例
    public static Singleton getInstance() {
        if (uniqueInstance == null) { // 如果 uniqueInstance 为空,表示还没有创建实例...
            // 通过构造器的私有方法实例化 Singleton,并赋值给 uniqueInstance
            // 注意,如果我们不需要这个实例,它就不会被创建,这就是延迟实例化(lazy instantiation)
            uniqueInstance = new Singleton(); 
        }
      // 如果 uniqueinstance 不为空,说明之前已经创建过对象,直接跳转到 return 语句
      return uniqueInstance;
    }
}


单例模式例子:特殊的计数器

我们可以写一个计数器,它的功能是用于保存它在程序执行期间被调用的次数。这个计数器的需要满足的几个要求:


  • 当之前没有创建过计数器 count 时,将创建一个新的计数器 count = 0
  • 如果已经创建了一个计数器,则返回此实例实际保存的 count
  • 如果我们调用方法 AddOne 一次,计数 count 必须增加 1


在这个场景下,我们需要有 3 个测试来坚持我们的单元测试。

第一个单元测试

与 Java 或 C++ 这种面向对象语言中不同,Go 实现单例模式没有像静态成员的东西(通过 static 修饰),但是可以通过包的范围来提供一个类似的功能。


首先,我们要为单例对象编写包的声明:

package singleton
type Singleton struct {
  count int
}
var instance *Singleton
func init() {
  instance = &Singleton{}
}
func GetInstance() *Singleton {
  return nil
}
func (s *Singleton) AddOne() int {
  return 0
}


然后,我们通过编写测试代码来验证我们声明的函数:

package singleton
import (
  "testing"
)
func TestGetInstance(t *testing.T) {
  count := GetInstance()
  if count == nil {
    t.Error("A new connection object must have been made")
  }
  expectedCounter := count
  currentCount := count.AddOne()
  if currentCount != 1 {
    t.Errorf("After calling for the first time to count, the count must be 1 but it is %d\n", currentCount)
  }
  count2 := GetInstance()
  if count2 != expectedCounter {
    t.Error("Singleton instances must be different")
  }
  currentCount = count2.AddOne()
  if currentCount != 2 {
    t.Errorf("After calling 'AddOne' using the second counter, the current count must be 2 but was %d\n", currentCount)
  }
}

第一个测试是检查是显而易见,但在复杂的应用中,其重要性也不小。当我们要求获得一个计数器的实例时,我们实际上需要得到一个结果。


我们把对象的创建委托给一个未知的包,而这个对象在创建或检索对象时可能失败。我们还将当前的计数器存储在变量 expectedCounter 中,以便以后进行比较。即:

  currentCount := count.AddOne()
  if currentCount != 1 {
    t.Errorf("After calling for the first time to count, the count must be 1 but it is %d\n", currentCount)
  }


运行上面的代码:

$ go test -v -run=GetInstance .
=== RUN   TestGetInstance
    singleton_test.go:12: A new connection object must have been made
    singleton_test.go:19: After calling for the first time to count, the count must be 1 but it is 0
    singleton_test.go:31: After calling 'AddOne' using the second counter, the current count must be 2 but was 0
--- FAIL: TestGetInstance (0.00s)
FAIL
FAIL    github.com/yuzhoustayhungry/GoDesignPattern/singleton   0.412s
FAIL
相关文章
|
3月前
|
程序员 Go 云计算
2023年学习Go语言是否值得?探索Go语言的魅力
2023年学习Go语言是否值得?探索Go语言的魅力
|
3月前
|
安全 测试技术 Go
|
3月前
|
缓存 NoSQL Go
通过 SingleFlight 模式学习 Go 并发编程
通过 SingleFlight 模式学习 Go 并发编程
|
6天前
|
设计模式 测试技术 Go
学习Go语言
【10月更文挑战第25天】学习Go语言
16 4
|
2月前
|
编译器 Go
go语言学习记录(关于一些奇怪的疑问)有别于其他编程语言
本文探讨了Go语言中的常量概念,特别是特殊常量iota的使用方法及其自动递增特性。同时,文中还提到了在声明常量时,后续常量可沿用前一个值的特点,以及在遍历map时可能遇到的非顺序打印问题。
|
3月前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
122 1
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
3月前
|
算法 NoSQL 中间件
go语言后端开发学习(六) ——基于雪花算法生成用户ID
本文介绍了分布式ID生成中的Snowflake(雪花)算法。为解决用户ID安全性与唯一性问题,Snowflake算法生成的ID具备全局唯一性、递增性、高可用性和高性能性等特点。64位ID由符号位(固定为0)、41位时间戳、10位标识位(含数据中心与机器ID)及12位序列号组成。面对ID重复风险,可通过预分配、动态或统一分配标识位解决。Go语言实现示例展示了如何使用第三方包`sonyflake`生成ID,确保不同节点产生的ID始终唯一。
go语言后端开发学习(六) ——基于雪花算法生成用户ID
|
3月前
|
Go
Go - 学习 grpc.Dial(target string, opts …DialOption) 的写法
Go - 学习 grpc.Dial(target string, opts …DialOption) 的写法
56 12
|
3月前
|
JSON 缓存 监控
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
Viper 是一个强大的 Go 语言配置管理库,适用于各类应用,包括 Twelve-Factor Apps。相比仅支持 `.ini` 格式的 `go-ini`,Viper 支持更多配置格式如 JSON、TOML、YAML
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
|
2月前
|
Rust Linux Go
Rust/Go语言学习
Rust/Go语言学习