Golang?sync.Once實(shí)現(xiàn)單例模式的方法詳解
Go 語(yǔ)言的 sync 包提供了一系列同步原語(yǔ),其中 sync.Once 就是其中之一。sync.Once 的作用是保證某個(gè)函數(shù)只會(huì)被執(zhí)行一次,即使在多個(gè) goroutine 中也不會(huì)重復(fù)執(zhí)行。sync.Once 在實(shí)際開(kāi)發(fā)中非常常用,例如在單例模式中。
本文將深入探討 sync.Once 的實(shí)現(xiàn)原理和使用方法,幫助大家更好地理解和應(yīng)用 sync.Once。
1. sync.Once 的原理和實(shí)現(xiàn)
Golang 的 sync.Once 是一個(gè)并發(fā)原語(yǔ),用于確保某個(gè)函數(shù)在整個(gè)程序運(yùn)行期間只會(huì)執(zhí)行一次。在內(nèi)部實(shí)現(xiàn)中,sync.Once 基于 sync.Mutex 和 sync.Cond,通過(guò)互斥鎖和條件變量來(lái)實(shí)現(xiàn)線程安全和防止重復(fù)執(zhí)行。下面是一個(gè)簡(jiǎn)單的示例:
package main
?
import (
"fmt"
"sync"
)
?
func main() {
var once sync.Once
?
// 保證只會(huì)執(zhí)行一次
once.Do(func() {
fmt.Println("Hello, World!")
})
}在這個(gè)示例中,我們使用 sync.Once 來(lái)確保 fmt.Println("Hello, World!") 只會(huì)執(zhí)行一次。如果我們多次調(diào)用 once.Do(),只有第一次會(huì)真正執(zhí)行,后續(xù)的調(diào)用都會(huì)直接返回。這種保證只執(zhí)行一次的機(jī)制非常適用于一些需要緩存結(jié)果、初始化狀態(tài)或者注冊(cè)回調(diào)函數(shù)等場(chǎng)景。
sync.Once 的實(shí)現(xiàn)基于兩個(gè)核心的概念:互斥鎖和條件變量。sync.Once 內(nèi)部維護(hù)了一個(gè)狀態(tài)標(biāo)志位 done,用于標(biāo)記函數(shù)是否已經(jīng)被執(zhí)行過(guò)。如果 done 的值為 true,那么 sync.Once 就認(rèn)為函數(shù)已經(jīng)執(zhí)行過(guò),后續(xù)的調(diào)用直接返回;如果 done 的值為 false,那么 sync.Once 就認(rèn)為函數(shù)還沒(méi)有執(zhí)行過(guò),然后通過(guò)互斥鎖和條件變量來(lái)保證函數(shù)的線程安全性和只執(zhí)行一次的特性。
sync.Once 是一個(gè)非常簡(jiǎn)單的類型,它只有一個(gè) Do 方法,下面是 sync.Once 的內(nèi)部實(shí)現(xiàn)代碼:
type Once struct {
m Mutex
done uint32
}
?
func (o *Once) Do(f func()) {
if atomic.LoadUint32(&o.done) == 1 {
return
}
?
o.m.Lock()
defer o.m.Unlock()
if o.done == 0 {
defer atomic.StoreUint32(&o.done, 1)
f()
}
}從上面的代碼可以看出,sync.Once 的實(shí)現(xiàn)非常簡(jiǎn)單。在 Do 方法中,它首先檢查 done 字段是否為 1,如果是,則直接返回,否則就獲取鎖。獲取鎖之后,它再次檢查 done 字段是否為 0,如果是,則執(zhí)行傳入的函數(shù) f,并將 done 字段設(shè)置為 1。由于只有一個(gè) goroutine 能夠獲取到鎖并執(zhí)行 f,所以 sync.Once 可以保證 f 只會(huì)被執(zhí)行一次。
需要注意的是,sync.Once 的實(shí)現(xiàn)中使用了 defer 關(guān)鍵字,這是為了保證在函數(shù)返回時(shí)能夠釋放鎖,并將 done 字段設(shè)置為 1。這種寫(xiě)法非常巧妙,能夠避免很多常見(jiàn)的并發(fā)問(wèn)題,比如死鎖、競(jìng)爭(zhēng)條件等。
2. sync.Once 的錯(cuò)誤處理
由于 sync.Once 能夠確保某個(gè)函數(shù)只會(huì)執(zhí)行一次,因此在函數(shù)執(zhí)行失敗時(shí),我們需要考慮如何處理錯(cuò)誤。
一種常見(jiàn)的錯(cuò)誤處理方式是將錯(cuò)誤信息存儲(chǔ)在 sync.Once 結(jié)構(gòu)體中,并在后續(xù)的調(diào)用中返回錯(cuò)誤信息。下面是一個(gè)示例:
package main
?
import (
"errors"
"fmt"
"sync"
)
?
type Config struct {
Name string
}
?
var (
config *Config
configOnce sync.Once
configErr error
)
?
func loadConfig() error {
// 模擬配置加載失敗
return errors.New("failed to load config")
}
?
func getConfig() (*Config, error) {
configOnce.Do(func() {
// 只有在第一次執(zhí)行時(shí)才會(huì)調(diào)用 loadConfig 函數(shù)
if err := loadConfig(); err != nil {
configErr = err
} else {
config = &Config{Name: "example"}
}
})
?
return config, configErr
}
?
func main() {
cfg, err := getConfig()
if err != nil {
fmt.Printf("error: %v\n", err)
return
}
?
fmt.Printf("config: %+v\n", cfg)
}在這個(gè)示例中,我們使用 sync.Once 來(lái)確保 getConfig() 函數(shù)只會(huì)執(zhí)行一次。在第一次執(zhí)行時(shí),我們通過(guò) loadConfig() 函數(shù)加載配置,如果加載失敗,我們將錯(cuò)誤信息存儲(chǔ)在 configErr 變量中,否則將配置信息存儲(chǔ)在 config 變量中。在后續(xù)的調(diào)用中,我們將 config 和 configErr 一起返回,這樣就能夠正確地處理函數(shù)執(zhí)行失敗的情況了。
3. sync.Once 的嵌套調(diào)用
在某些情況下,我們可能需要在 sync.Once 中嵌套調(diào)用其他函數(shù),以實(shí)現(xiàn)更復(fù)雜的邏輯。這時(shí)候我們需要注意的是,在嵌套調(diào)用中,我們需要使用新的 sync.Once 實(shí)例來(lái)保證內(nèi)部函數(shù)的執(zhí)行只會(huì)發(fā)生一次。下面是一個(gè)示例:
package main
?
import (
"fmt"
"sync"
)
?
func main() {
var once sync.Once
?
// 外層函數(shù)
outer := func() {
fmt.Println("outer")
// 內(nèi)層函數(shù)
inner := func() {
fmt.Println("inner")
}
?
var innerOnce sync.Once
innerOnce.Do(inner)
}
?
// 外層函數(shù)只會(huì)執(zhí)行一次
once.Do(outer)
once.Do(outer)
}在這個(gè)示例中,我們定義了一個(gè)外層函數(shù) outer 和一個(gè)內(nèi)層函數(shù) inner,然后在 outer 函數(shù)中使用了一個(gè)新的 sync.Once 實(shí)例 innerOnce 來(lái)保證 inner 函數(shù)只會(huì)執(zhí)行一次。在最后的調(diào)用中,我們使用一個(gè)新的 sync.Once 實(shí)例 once 來(lái)保證 outer 函數(shù)只會(huì)執(zhí)行一次,避免了重復(fù)執(zhí)行造成的問(wèn)題。
4. 并發(fā)性能
在并發(fā)編程中,性能是一個(gè)非常重要的指標(biāo)。因此,我們需要了解 sync.Once 在并發(fā)場(chǎng)景下的性能表現(xiàn),以便在實(shí)際應(yīng)用中選擇合適的并發(fā)控制方案。
sync.Once 的性能表現(xiàn)在很大程度上取決于被保護(hù)函數(shù)的實(shí)際執(zhí)行時(shí)間。如果被保護(hù)函數(shù)執(zhí)行時(shí)間很長(zhǎng),那么 sync.Once 的性能表現(xiàn)會(huì)受到影響,因?yàn)槊總€(gè) goroutine 都需要等待被保護(hù)函數(shù)的執(zhí)行結(jié)束才能繼續(xù)執(zhí)行。
下面是一個(gè)簡(jiǎn)單的性能測(cè)試示例,用于比較 sync.Once 和傳統(tǒng)的鎖機(jī)制在并發(fā)場(chǎng)景下的性能表現(xiàn):
package main
?
import (
"sync"
"sync/atomic"
"time"
)
?
const (
numGoroutines = 1000
numRepeats = 100
)
?
func testWithSyncOnce() {
var once sync.Once
?
for i := 0; i < numGoroutines; i++ {
go func() {
for j := 0; j < numRepeats; j++ {
once.Do(func() {
time.Sleep(10 * time.Millisecond)
})
}
}()
}
}
?
func testWithMutex() {
var mutex sync.Mutex
var done int64
?
for i := 0; i < numGoroutines; i++ {
go func() {
for j := 0; j < numRepeats; j++ {
mutex.Lock()
if done == 0 {
time.Sleep(10 * time.Millisecond)
atomic.StoreInt64(&done, 1)
}
mutex.Unlock()
}
}()
}
}
?
func main() {
start := time.Now()
testWithSyncOnce()
fmt.Printf("sync.Once: %v\n", time.Since(start))
?
start = time.Now()
testWithMutex()
fmt.Printf("Mutex: %v\n", time.Since(start))
}在這個(gè)示例中,我們定義了兩個(gè)函數(shù) testWithSyncOnce 和 testWithMutex,分別使用 sync.Once 和傳統(tǒng)的鎖機(jī)制來(lái)實(shí)現(xiàn)并發(fā)控制。在每個(gè)函數(shù)中,我們使用 numGoroutines 個(gè) goroutine 來(lái)執(zhí)行被保護(hù)函數(shù),并重復(fù)執(zhí)行 numRepeats 次。
在 main 函數(shù)中,我們使用 time 包來(lái)測(cè)量?jī)蓚€(gè)函數(shù)的執(zhí)行時(shí)間,并比較它們的性能表現(xiàn)。
實(shí)際上,由于 sync.Once 內(nèi)部使用原子操作來(lái)控制執(zhí)行狀態(tài),因此在被保護(hù)函數(shù)執(zhí)行時(shí)間很短的情況下,sync.Once 的性能表現(xiàn)要優(yōu)于傳統(tǒng)的鎖機(jī)制。但是,在被保護(hù)函數(shù)執(zhí)行時(shí)間較長(zhǎng)的情況下,sync.Once 的性能表現(xiàn)會(huì)逐漸變差。
在實(shí)際應(yīng)用中,我們需要根據(jù)被保護(hù)函數(shù)的實(shí)際執(zhí)行時(shí)間和并發(fā)訪問(wèn)量來(lái)選擇合適的并發(fā)控制方案。
5. 總結(jié)
在本文中,我們介紹了 sync.Once 的基本用法,并討論了它在錯(cuò)誤處理、嵌套調(diào)用和并發(fā)性能方面的注意事項(xiàng)。在實(shí)際應(yīng)用中,sync.Once 是一個(gè)非常實(shí)用的并發(fā)控制工具,它可以保證某個(gè)函數(shù)只會(huì)執(zhí)行一次,并提高程序的性能表現(xiàn)。
當(dāng)然,除了 sync.Once,Golang 還提供了其他的并發(fā)控制工具,比如 sync.WaitGroup、sync.Mutex 等。在實(shí)際應(yīng)用中,我們需要根據(jù)具體的場(chǎng)景來(lái)選擇合適的并發(fā)控制方案。
如果被保護(hù)函數(shù)的執(zhí)行時(shí)間很短,且并發(fā)訪問(wèn)量較高,那么可以考慮使用 sync.Once。如果被保護(hù)函數(shù)的執(zhí)行時(shí)間較長(zhǎng),或者需要多個(gè) goroutine 共同協(xié)作完成某個(gè)任務(wù),那么可以考慮使用 sync.WaitGroup。如果需要保證某些資源在同一時(shí)刻只能被一個(gè) goroutine 訪問(wèn),那么可以考慮使用 sync.Mutex。
在實(shí)際應(yīng)用中,我們需要根據(jù)具體的場(chǎng)景來(lái)選擇合適的并發(fā)控制方案,并結(jié)合實(shí)際性能測(cè)試數(shù)據(jù)來(lái)進(jìn)行優(yōu)化。只有合理使用并發(fā)控制工具,才能充分發(fā)揮 Golang 的并發(fā)編程優(yōu)勢(shì),提高程序的性能和穩(wěn)定性。
以上就是Golang sync.Once實(shí)現(xiàn)單例模式的方法詳解的詳細(xì)內(nèi)容,更多關(guān)于Golang sync.Once的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go?常見(jiàn)設(shè)計(jì)模式之單例模式詳解
單例模式是設(shè)計(jì)模式中最簡(jiǎn)單的一種模式,單例模式能夠確保無(wú)論對(duì)象被實(shí)例化多少次,全局都只有一個(gè)實(shí)例存在,在Go?語(yǔ)言有多種方式可以實(shí)現(xiàn)單例模式,所以我們今天就來(lái)一起學(xué)習(xí)下吧2023-07-07
如何在Go中使用Casbin進(jìn)行訪問(wèn)控制
這篇文章主要介紹了如何在Go中使用Casbin進(jìn)行訪問(wèn)控制,Casbin是一個(gè)強(qiáng)大的、高效的開(kāi)源訪問(wèn)控制框架,其權(quán)限管理機(jī)制支持多種訪問(wèn)控制模型,Casbin只負(fù)責(zé)訪問(wèn)控制2022-08-08
詳解go程序如何在windows服務(wù)中開(kāi)啟和關(guān)閉
這篇文章主要介紹了一個(gè)go程序,如何在windows服務(wù)中優(yōu)雅開(kāi)啟和關(guān)閉,文中通過(guò)代碼示例和圖文講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-07-07
Golang 使用接口實(shí)現(xiàn)泛型的方法示例
這篇文章主要介紹了Golang 使用接口實(shí)現(xiàn)泛型的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
淺談Go語(yǔ)言中的結(jié)構(gòu)體struct & 接口Interface & 反射
下面小編就為大家?guī)?lái)一篇淺談Go語(yǔ)言中的結(jié)構(gòu)體struct & 接口Interface & 反射。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07
Go數(shù)據(jù)庫(kù)遷移的實(shí)現(xiàn)步驟
本文主要介紹了Go數(shù)據(jù)庫(kù)遷移的實(shí)現(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07

