Go語(yǔ)言單例模式詳解
單例模式是一種常見(jiàn)的設(shè)計(jì)模式,它在系統(tǒng)中僅允許創(chuàng)建一個(gè)實(shí)例來(lái)控制對(duì)某些資源的訪問(wèn)。在 Go 語(yǔ)言中,實(shí)現(xiàn)單例模式有多種方式,本篇文章將帶你深入掌握 Go 語(yǔ)言中的單例模式實(shí)現(xiàn)。
什么是單例模式
單例模式指僅允許創(chuàng)建一個(gè)對(duì)象的設(shè)計(jì)模式。它通常應(yīng)用于控制對(duì)某些資源的訪問(wèn),例如數(shù)據(jù)庫(kù)連接、線程池等等。通過(guò)單例模式,可以確保系統(tǒng)中只存在唯一一個(gè)實(shí)例,并提供一個(gè)全局訪問(wèn)點(diǎn),方便其他對(duì)象使用。
Go中實(shí)現(xiàn)單例模式的方式
Go 語(yǔ)言提供了多種方式來(lái)實(shí)現(xiàn)單例模式,包括以下幾種方法:
方法一: 懶漢式
懶漢式是一種常見(jiàn)的單例模式實(shí)現(xiàn)方式,其特點(diǎn)是在首次使用時(shí)創(chuàng)建單例實(shí)例。實(shí)現(xiàn)方法如下:
package singleton
import "sync"
var (
instance *Singleton
once sync.Once
)
type Singleton struct {
}
func GetInstance() *Singleton {
once.Do(func() {
instance = &Singleton{}
})
return instance
}在上面的代碼中,我們定義了一個(gè)名為 Singleton 的結(jié)構(gòu)體,并將其實(shí)例化為 instance。同時(shí),使用 sync 包中的 sync.Once 對(duì)象 once 實(shí)現(xiàn)在程序生命周期內(nèi)只執(zhí)行一次的邏輯。當(dāng)?shù)谝淮握{(diào)用 GetInstance 時(shí),Sync.Once 的 Do 方法會(huì)調(diào)用傳入的函數(shù),該函數(shù)的邏輯是實(shí)例化 Singleton 對(duì)象并賦值給 instance。在后續(xù)調(diào)用 GetInstance 時(shí),由于 instance 已經(jīng)被實(shí)例化,不會(huì)再次創(chuàng)建。
方法二:餓漢式
餓漢式是另一種常見(jiàn)的單例模式實(shí)現(xiàn)方式,其特點(diǎn)是在系統(tǒng)啟動(dòng)時(shí)即創(chuàng)建單例實(shí)例,當(dāng)調(diào)用時(shí)直接返回該實(shí)例。實(shí)現(xiàn)方法如下:
package singleton
var instance *Singleton = &Singleton{}
type Singleton struct {
}
func GetInstance() *Singleton {
return instance
}
在上面的代碼中,我們?cè)诎跏蓟瘯r(shí)創(chuàng)建了一個(gè) Singleton 對(duì)象并賦值為 instance。GetInstace 方法直接返回該實(shí)例,因此每一次調(diào)用都返回同一個(gè)對(duì)象,達(dá)到了控制對(duì)象實(shí)例的目的。
方法三:雙重檢查鎖定
雙重檢查鎖定是一種在多線程環(huán)境下使用的單例模式實(shí)現(xiàn)方式,其特點(diǎn)是先檢查是否已經(jīng)有實(shí)例,如果沒(méi)有則進(jìn)入同步代碼塊進(jìn)行創(chuàng)建。實(shí)現(xiàn)方法如下:
package singleton
import "sync"
var (
instance *Singleton
mu sync.Mutex
)
type Singleton struct {
}
func GetInstance() *Singleton {
if instance == nil {
mu.Lock()
defer mu.Unlock()
if instance == nil {
instance = &Singleton{}
}
}
return instance
}
在上面的代碼中,我們使用了互斥鎖實(shí)現(xiàn)并發(fā)控制,確保在多線程環(huán)境下只有一個(gè)線程能夠訪問(wèn)臨界資源。同時(shí),使用了一個(gè)雙重檢查機(jī)制,減少互斥鎖使用的頻率。
示例
通過(guò)下面的示例,我們可以了解如何使用單例模式來(lái)實(shí)現(xiàn)數(shù)據(jù)緩存。
package main
import (
"fmt"
"sync"
)
type Cache struct {
store map[string]string
mu sync.Mutex
}
var instance *Cache
func GetCacheInstance() *Cache {
if instance == nil {
instance = &Cache{
store: make(map[string]string),
}
}
return instance
}
func (c *Cache) Get(key string) (string, bool) {
c.mu.Lock()
defer c.mu.Unlock()
val, ok := c.store[key]
return val, ok
}
func (c *Cache) Set(key, val string) {
c.mu.Lock()
defer c.mu.Unlock()
c.store[key] = val
}
func main() {
cache := GetCacheInstance()
cache.Set("name", "Tom")
if val, ok := cache.Get("name"); ok {
fmt.Println(val)
}
}
在上面的代碼中,我們定義了一個(gè) Cache 結(jié)構(gòu)體,表示數(shù)據(jù)緩存。Cache 中包含了 Store 成員變量用于存儲(chǔ)鍵值對(duì),使用互斥鎖 mu 控制訪問(wèn)。GetCacheInstance 函數(shù)返回一個(gè) Cache 實(shí)例,表示數(shù)據(jù)緩存,在第一次調(diào)用時(shí),會(huì)將 instance 實(shí)例化為一個(gè) Cache 對(duì)象。示例中的 main 函數(shù)演示了如何使用單例的 Cache 對(duì)象來(lái)存儲(chǔ)和獲取數(shù)據(jù)。
總結(jié)
單例模式是一種常見(jiàn)的設(shè)計(jì)模式,它確保在系統(tǒng)中只存在唯一一個(gè)實(shí)例,并提供一個(gè)全局訪問(wèn)點(diǎn)。在 Go 語(yǔ)言中,實(shí)現(xiàn)單例模式有多種方式,包括懶漢式、餓漢式、雙重檢查鎖定等。通過(guò)本文,我們了解了這幾種實(shí)現(xiàn)方法的具體細(xì)節(jié),并實(shí)現(xiàn)了一個(gè)數(shù)據(jù)緩存緩存的示例。
到此這篇關(guān)于Go語(yǔ)言單例模式詳解的文章就介紹到這了,更多相關(guān)Go語(yǔ)言單例模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go外部依賴包從vendor,$GOPATH和$GOPATH/pkg/mod查找順序
這篇文章主要介紹了Go外部依賴包vendor,$GOPATH和$GOPATH/pkg/mod下查找順序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12
Go語(yǔ)言net包RPC遠(yuǎn)程調(diào)用三種方式http與json-rpc及tcp
這篇文章主要為大家介紹了Go語(yǔ)言net包RPC遠(yuǎn)程調(diào)用三種方式分別使用http與json-rpc及tcp的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-11-11
解決Golang小數(shù)float64在實(shí)際工程中加減乘除的精度問(wèn)題
這篇文章主要介紹了解決Golang小數(shù)float64在實(shí)際工程中加減乘除的精度問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-03-03
簡(jiǎn)單高效!Go語(yǔ)言封裝二級(jí)認(rèn)證功能實(shí)現(xiàn)
本文將介紹如何使用Go語(yǔ)言封裝二級(jí)認(rèn)證功能,實(shí)現(xiàn)簡(jiǎn)單高效的用戶認(rèn)證流程,二級(jí)認(rèn)證是一種安全措施,要求用戶在登錄后進(jìn)行額外的身份驗(yàn)證,以提高賬戶安全性,2023-10-10
golang gc的內(nèi)部?jī)?yōu)化詳細(xì)介紹
Go編譯器在垃圾回收(GC)的掃描標(biāo)記階段,對(duì)存儲(chǔ)無(wú)指針鍵值對(duì)的map進(jìn)行了優(yōu)化,即在GC掃描時(shí)不深入掃描map內(nèi)部數(shù)據(jù),只檢查map本身是否需要回收,這一優(yōu)化顯著提升了GC掃描的速度,從而減少了GC對(duì)程序性能的影響2024-10-10

