欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go?常見設(shè)計(jì)模式之單例模式詳解

 更新時間:2023年07月06日 10:23:26   作者:江湖十年  
單例模式是設(shè)計(jì)模式中最簡單的一種模式,單例模式能夠確保無論對象被實(shí)例化多少次,全局都只有一個實(shí)例存在,在Go?語言有多種方式可以實(shí)現(xiàn)單例模式,所以我們今天就來一起學(xué)習(xí)下吧

餓漢式

餓漢式實(shí)現(xiàn)單例模式非常簡單,直接看代碼:

package singleton
type singleton struct{}
var instance = &singleton{}
func GetSingleton() *singleton {
	return instance
}

singleton 包在被導(dǎo)入時會自動初始化 instance 實(shí)例,使用時通過調(diào)用 singleton.GetSingleton() 函數(shù)即可獲得 singleton 這個結(jié)構(gòu)體的單例對象。

由于單例對象是在包加載時立即被創(chuàng)建出來,所以也就有了這個形象的名稱叫作餓漢式。與之對應(yīng)的另一種實(shí)現(xiàn)方式叫作懶漢式,當(dāng)實(shí)例被第一次使用時才會被創(chuàng)建。

需要注意的是,盡管餓漢式實(shí)現(xiàn)單例模式如此簡單,但大多數(shù)情況下仍不被推薦使用,因?yàn)槿绻麊卫龑?shí)例化時初始化內(nèi)容過多,可能造成程序加載用時較長。

懶漢式

接下來我們再來看下如何通過懶漢式實(shí)現(xiàn)單例模式:

package singleton
type singleton struct{}
var instance *singleton
func GetSingleton() *singleton {
	if instance == nil {
		instance = &singleton{}
	}
	return instance
}

相較于餓漢式的實(shí)現(xiàn),我們把實(shí)例化 singleton 結(jié)構(gòu)體部分的代碼移到了 GetSingleton() 函數(shù)內(nèi)部。這樣一來,就將對象實(shí)例化的步驟延遲到了 GetSingleton() 被第一次調(diào)用時。

通過 instance == nil 的判斷來實(shí)現(xiàn)單例并不十分可靠,當(dāng)有多個 goroutine 同時調(diào)用 GetSingleton() 時無法保證并發(fā)安全。

支持并發(fā)的單例

如果你用 Go 語言寫過并發(fā)編程,那么應(yīng)該可以很快想到解決懶漢式單例模式并發(fā)安全問題的方案:

package singleton
import "sync"
type singleton struct{}
var instance *singleton
var mu sync.Mutex
func GetSingleton() *singleton {
	mu.Lock()
	defer mu.Unlock()
	if instance == nil {
		instance = &singleton{}
	}
	return instance
}

我們對代碼的主要修改就是在 GetSingleton() 函數(shù)最開始加了如下兩行代碼:

mu.Lock()
defer mu.Unlock()

通過加鎖的機(jī)制,就可以保證這個實(shí)現(xiàn)單例模式的函數(shù)是并發(fā)安全的。

不過這樣也有些問題,因?yàn)橛昧随i機(jī)制,每次調(diào)用 GetSingleton() 時程序都會進(jìn)行加鎖、解鎖的步驟,這樣會導(dǎo)致程序性能的下降。

雙重鎖定

加鎖導(dǎo)致程序性能下降,但我們又不得不用鎖來保證程序的并發(fā)安全,于是有人想出了雙重鎖定(Double-Check Locking)的方案:

package singleton
import "sync"
type singleton struct{}
var instance *singleton
var mu sync.Mutex
func GetSingleton() *singleton {
	if instance == nil {
		mu.Lock()
		defer mu.Unlock()
		if instance == nil {
			instance = &singleton{}
		}
	}
	return instance
}

可以看到,所謂的雙重鎖定實(shí)際上就是在程序加鎖前又加了一層 instance == nil 判斷,這樣就兼顧了性能和安全兩個方面。

不過這段代碼看起來有些奇怪,既然外層已經(jīng)判斷了 instance == nil,加鎖后卻又進(jìn)行了第二次 instance == nil 判斷。其實(shí)外層的 instance == nil 判斷是為了提高程序的執(zhí)行效率,因?yàn)槿绻?instance 已經(jīng)存在,則無需進(jìn)入 if 邏輯,程序直接返回 instance 即可。這樣就免去了原來每次調(diào)用 GetSingleton() 都上鎖的操作,將加鎖的粒度更加精細(xì)化。而內(nèi)層的 instance == nil 判斷則是考慮了并發(fā)安全,在極端情況下,多個 goroutine 同時走到了加鎖這一步,內(nèi)層判斷就起到作用了。

Gopher 慣用方案

雖然我們通過雙重鎖定機(jī)制兼顧和性能和并發(fā)安全,但代碼有些丑陋,不符合廣大 Gopher 的期待。好在 Go 語言在 sync 包中提供了 Once 機(jī)制能夠幫助我們寫出更加優(yōu)雅的代碼:

package singleton
import "sync"
type singleton struct{}
var instance *singleton
var once sync.Once
func GetSingleton() *singleton {
	once.Do(func() {
		instance = &singleton{}
	})
	return instance
}

Once 是一個結(jié)構(gòu)體,在執(zhí)行 Do 方法的內(nèi)部通過 atomic 操作和加鎖機(jī)制來保證并發(fā)安全,且 once.Do 能夠保證多個 goroutine 同時執(zhí)行時 &singleton{} 只被創(chuàng)建一次。

其實(shí) Once 并不神秘,其內(nèi)部實(shí)現(xiàn)跟上面使用的雙重鎖定機(jī)制非常類似,只不過把 instance == nil 換成了 atomic 操作,感興趣的同學(xué)可以查看下其對應(yīng)源碼。

總結(jié)

以上就是 Go 語言中實(shí)現(xiàn)單例模式的幾種常用套路,經(jīng)過對比可以得出結(jié)論,最推薦的方式是使用 once.Do 來實(shí)現(xiàn),sync.Once 包幫我們隱藏了部分細(xì)節(jié),卻可以讓代碼可讀性得到很大提升。

希望此文能對你有所幫助。

到此這篇關(guān)于Go 常見設(shè)計(jì)模式之單例模式詳解的文章就介紹到這了,更多相關(guān)Go單例模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go獲取與設(shè)置環(huán)境變量的方法詳解

    Go獲取與設(shè)置環(huán)境變量的方法詳解

    go環(huán)境變量的配置其實(shí)說真的說難也難,說不難也不難,只要配置成功過一次以后后面都很簡單,但是最好是要做好筆記,這篇文章主要給大家介紹了關(guān)于Go獲取與設(shè)置環(huán)境變量的相關(guān)資料,需要的朋友可以參考下
    2021-11-11
  • Golang中的godoc使用簡介(推薦)

    Golang中的godoc使用簡介(推薦)

    Godoc是go語言的文檔化工具,類似于文檔化工具godoc,類似于Python的Docstring和Java的Javadoc,這篇文章主要介紹了Golang中的godoc使用簡介,需要的朋友可以參考下
    2022-10-10
  • GoLang中拼接字符串性能優(yōu)化方法詳解

    GoLang中拼接字符串性能優(yōu)化方法詳解

    最近在做性能優(yōu)化,有個函數(shù)里面的耗時特別長,看里面的操作大多是一些字符串拼接的操作,而字符串拼接在 golang 里面其實(shí)有很多種實(shí)現(xiàn),下面這篇文章主要給大家介紹了關(guān)于Golang語言如何高效拼接字符串的相關(guān)資料,需要的朋友可以參考下
    2023-02-02
  • Golang import本地包和導(dǎo)入問題相關(guān)詳解

    Golang import本地包和導(dǎo)入問題相關(guān)詳解

    這篇文章主要介紹了Golang import本地包和導(dǎo)入問題相關(guān)詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02
  • Go語言基礎(chǔ)go build命令用法及示例詳解

    Go語言基礎(chǔ)go build命令用法及示例詳解

    這篇文章主要為大家介紹了Go語言基礎(chǔ)go build命令用法及示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2021-11-11
  • golang sync.Pool 指針數(shù)據(jù)覆蓋問題解決

    golang sync.Pool 指針數(shù)據(jù)覆蓋問題解決

    本文主要介紹了使用sync.Pool時遇到指針數(shù)據(jù)覆蓋的問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2025-03-03
  • golang中package?is?not?in?GOROOT報(bào)錯的真正解決辦法

    golang中package?is?not?in?GOROOT報(bào)錯的真正解決辦法

    這篇文章主要給大家介紹了關(guān)于golang中package?is?not?in?GOROOT報(bào)錯的真正解決辦法,文中通過圖文介紹的非常詳細(xì),對同樣遇到這個問題的朋友具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2023-03-03
  • 深入了解Golang為什么需要超時控制

    深入了解Golang為什么需要超時控制

    本文將介紹為什么需要超時控制,然后詳細(xì)介紹Go語言中實(shí)現(xiàn)超時控制的方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下
    2023-05-05
  • Go語言使用GORM操作數(shù)據(jù)庫使用指南

    Go語言使用GORM操作數(shù)據(jù)庫使用指南

    GORM(全稱為Go?Object?Relational?Mapping)是一個在Go語言中使用的輕量級的對象關(guān)系映射(ORM)庫,本文主要為大家介紹了GORM操作數(shù)據(jù)庫具體方法,需要的可以參考一下
    2023-05-05
  • golang高并發(fā)系統(tǒng)限流策略漏桶和令牌桶算法源碼剖析

    golang高并發(fā)系統(tǒng)限流策略漏桶和令牌桶算法源碼剖析

    這篇文章主要介紹了golang高并發(fā)系統(tǒng)限流策略漏桶和令牌桶算法源碼剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06

最新評論