Go语言优雅实现单例模式的多种方式

 更新时间:2025年02月06日 08:20:36   作者:烛阴  
单例模式(Singleton Pattern)是一种设计模式,旨在保证一个类只有一个实例,并且提供全局访问点,单例模式通常用于需要限制某个对象的实例数量为一个的场景,本文给大家介绍了Go语言实现单例模式的多种方式,需要的朋友可以参考下

单例模式的基本定义

单例模式(Singleton Pattern)是一种设计模式,旨在保证一个类只有一个实例,并且提供全局访问点。单例模式通常用于需要限制某个对象的实例数量为一个的场景,比如数据库连接池、日志管理器、配置管理器等。

Go语言单例模式的实现

1. 线程安全的懒汉式单例

懒汉式的实现会延迟实例的创建,直到第一次调用时才会实例化对象。为了保证并发情况下的安全性,我们需要使用 sync.Once 来确保实例只会创建一次。

package main

import (
	"fmt"
	"sync"
)

var wg sync.WaitGroup

// Singleton 类型
type Singleton struct {
}

var instance *Singleton
var once sync.Once

// GetInstance 提供全局唯一的实例
func GetInstance() *Singleton {

	once.Do(func() {
		instance = &Singleton{}
	})
	return instance
}

func main() {
	// 获取单例实例
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(index int) {
			defer wg.Done()
			s1 := GetInstance()
			fmt.Printf("index %d, memery address: %p\n", index, s1)
		}(i)

	}
	wg.Wait()
}

结果

index 0, memery address: 0x56c480
index 5, memery address: 0x56c480
index 4, memery address: 0x56c480
index 2, memery address: 0x56c480
index 7, memery address: 0x56c480
index 9, memery address: 0x56c480
index 6, memery address: 0x56c480
index 8, memery address: 0x56c480
index 3, memery address: 0x56c480
index 1, memery address: 0x56c480

解析:

  • sync.Once:Go标准库提供的一个同步原语,确保某个函数只会被调用一次。它在并发情况下保证了线程安全。
  • once.Do:此方法确保传入的函数只执行一次,适用于懒加载单例实例。

2. 双重检查锁定(DCL)

双重检查锁定是一种优化方式,它通过在两次检查实例时,减少了加锁的开销,提高了性能。

package main

import (
	"fmt"
	"sync"
)

var wg sync.WaitGroup

// Singleton 类型
type Singleton struct {
}

var instance *Singleton
var lock sync.Mutex

func GetInstance() *Singleton {
	if instance == nil {
		lock.Lock()
		defer lock.Unlock()
		if instance == nil {
			instance = &Singleton{}
		}
	}
	return instance
}

func main() {
	// 获取单例实例
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(index int) {
			defer wg.Done()
			s1 := GetInstance()
			fmt.Printf("index %d, memery address: %p\n", index, s1)
		}(i)

	}
	wg.Wait()
}

解析:

  • 双重检查:第一次检查实例是否为 nil,如果是,则加锁。然后再次检查实例是否为 nil,如果是则创建实例。
  • 锁的优化:只有在实例尚未创建时才会加锁,避免了每次获取实例时都需要加锁的性能损耗。

3. 原子操作法

Go 语言的 sync/atomic 包提供了原子操作,我们可以利用它来确保单例的线程安全。

package main

import (
	"fmt"
	"sync"
	"sync/atomic"
	"unsafe"
)

var wg sync.WaitGroup

type Singleton struct {
}

var instance unsafe.Pointer

func GetInstance() *Singleton {
	// 使用原子操作获取实例
	if atomic.LoadPointer(&instance) == nil {
		newInstance := &Singleton{}
		atomic.StorePointer(&instance, unsafe.Pointer(newInstance))
	}
	return (*Singleton)(atomic.LoadPointer(&instance))
}

func main() {
	// 获取单例实例
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(index int) {
			defer wg.Done()
			s1 := GetInstance()
			fmt.Printf("index %d, memery address: %p\n", index, s1)
		}(i)

	}
	wg.Wait()
}

解析:

  • unsafe.Pointer:在Go中,unsafe.Pointer 可以用来绕过类型系统,直接处理内存地址。通过原子操作确保实例赋值的安全性。
  • atomic.LoadPointeratomic.StorePointer:原子加载和存储指针,确保操作的线程安全性。

总结

在Go语言中实现单例模式有多种方式,最常见的是使用 sync.Once、双重检查锁定(DCL)和原子操作法。每种方法有其优缺点,选择合适的方式可以帮助你在保证线程安全的前提下优化性能。

到此这篇关于Go语言优雅实现单例模式的多种方式的文章就介绍到这了,更多相关Go实现单例模式内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解Go语言中的数据类型及类型转换

    详解Go语言中的数据类型及类型转换

    这篇文章主要为大家介绍了Go语言中常见的几种数据类型,以及他们之间的转换方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-04-04
  • Golang websocket协议使用浅析

    Golang websocket协议使用浅析

    这篇文章主要介绍了Golang websocket协议的使用,WebSocket是一种新型的网络通信协议,可以在Web应用程序中实现双向通信,感兴趣想要详细了解可以参考下文
    2023-05-05
  • 使用Golang采集Nginx接口流量大小的步骤

    使用Golang采集Nginx接口流量大小的步骤

    在开发和运维中,我们经常需要监控和分析服务器的接口流量大小,特别是对于部署了 Nginx 的服务器,本文将介绍如何使用 Golang 采集 Nginx 接口流量大小,并展示如何将这些数据进行实时监控和分析
    2023-11-11
  • Golang并发编程中Context包的使用与并发控制

    Golang并发编程中Context包的使用与并发控制

    Golang的context包提供了在并发编程中传递取消信号、超时控制和元数据的功能,本文就来介绍一下Golang并发编程中Context包的使用与并发控制,感兴趣的可以了解一下
    2024-11-11
  • golang gorm更新日志执行SQL示例详解

    golang gorm更新日志执行SQL示例详解

    这篇文章主要为大家介绍了golang gorm更新日志执行SQL示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2022-04-04
  • 使用Go http重试请求的示例

    使用Go http重试请求的示例

    开发中对于http请求是经常遇到,一般可能网络延迟或接口返回超时,这篇文章主要介绍了使用Go http重试请求的示例,需要的朋友可以参考下
    2022-08-08
  • go中实现字符切片和字符串互转

    go中实现字符切片和字符串互转

    这篇文章主要为大家详细介绍了go语言中如何实现字符切片和字符串互转,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-11-11
  • Golang 中判断两个结构体相等的方法

    Golang 中判断两个结构体相等的方法

    这篇文章主要介绍了Golang 中如何判断两个结构体相等,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-08-08
  • Gin golang web开发模型绑定实现过程解析

    Gin golang web开发模型绑定实现过程解析

    这篇文章主要介绍了Gin golang web开发模型绑定实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • Golang error使用场景介绍

    Golang error使用场景介绍

    我们在使用Golang时,不可避免会遇到异常情况的处理,与Java、Python等语言不同的是,Go中并没有try...catch...这样的语句块,这个时候我们如何才能更好的处理异常呢?本文来教你正确方法
    2023-03-03

最新评论