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

golang使用sync.Once實現(xiàn)懶加載的用法和坑點詳解

 更新時間:2023年11月05日 13:55:16   作者:utmhikari  
這篇文章主要為大家詳細(xì)介紹了golang使用sync.Once實現(xiàn)懶加載的用法和坑點,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

在使用Golang做后端開發(fā)的工程中,我們通常需要聲明一些一些配置類或服務(wù)單例等在業(yè)務(wù)邏輯層面較為底層的實例。為了節(jié)省內(nèi)存或是冷啟動開銷,我們通常采用lazy-load懶加載的方式去初始化這些實例。初始化單例這個行為是一個非常經(jīng)典的并發(fā)處理的案例,比如在java當(dāng)中,我們可能用到建立雙重鎖+volatile的方式保證初始化邏輯只被訪問一次,并且所有線程最終都可以讀取到初始化完成的實例產(chǎn)物。這段經(jīng)典的代碼可以按如下的方式編寫:

public class Singleton {
    private volatile static Singleton uniqueSingleton;
 
    private Singleton() {
    }
 
    public Singleton getInstance() {
        if (null == uniqueSingleton) {
            synchronized (Singleton.class) {
                if (null == uniqueSingleton) {
                    uniqueSingleton = new Singleton();
                }
            }
        }
        return uniqueSingleton;
    }
}

但在Golang里面,實現(xiàn)懶加載的方式可以簡單的多,用內(nèi)置的sync.Once就能滿足。假設(shè)我們有一個user單例,需要被1000個線程讀取并打印,就可以這樣子寫:

type User struct {
    Name string `json:"name"`
    Age  int    `json:"age"`
}

var user *User
var userOnce sync.Once

func initUser() {
    user = &User{}
    cfgStr := `{"name":"foobar","age":18}`
    if err := json.Unmarshal([]byte(cfgStr), user); err != nil {
        panic("load user err: " + err.Error())
    }
}

func getUser() *User {
    userOnce.Do(initUser)
    return user
}

func TestSyncOnce(t *testing.T) {
    var wg sync.WaitGroup
    for i := 1; i < 1000; i++ {
        wg.Add(1)
        go func(n int) {
            defer wg.Done()
            curUser := getUser()
            t.Logf("[%d] got user: %+v", n, curUser)
        }(i)
    }
    wg.Wait()
}

這段代碼里,首先是通過var userOnce sync.Once聲明了一個sync.Once實例,然后在getUser當(dāng)中,我們聲明了userOnce.Do(initUser)這個操作。假設(shè)一個goroutine最先到達(dá)這個操作,就會上鎖并執(zhí)行initUser,其它goroutine到達(dá)之后,得等第一個goroutine執(zhí)行完initUser之后,才會繼續(xù)return user。這樣,就能一來保證initUser只會執(zhí)行一次,二來所有g(shù)oroutine都能夠最終讀到初始化完成的user單例。

sync.Once的工作機理也很簡單,通過一個鎖和一個flag就能夠?qū)崿F(xiàn):

func (o *Once) Do(f func()) {
	if atomic.LoadUint32(&o.done) == 0 { // 如果是1表示已經(jīng)完成了,跳過
		o.doSlow(f)
	}
}

func (o *Once) doSlow(f func()) {
	o.m.Lock() // 只有1個goroutine能拿到鎖,其它的等待
	defer o.m.Unlock()
	if o.done == 0 { // 如果還是0表示第一個來的,不是0就表示已經(jīng)有g(shù)oroutine做完了
		defer atomic.StoreUint32(&o.done, 1)
		f()
	}
}

最后也需要注意,sync.Once使用上面有一個坑點,不能也不需要像java一樣為單例提前做nil判斷。比如下面一段代碼是有問題的:

func initUser() {
    user = &User{} // 先給一個zero-value實例
    cfgStr := `{"name":"foobar","age":18}` // 然后加載json內(nèi)容,完成初始化
    if err := json.Unmarshal([]byte(cfgStr), user); err != nil {
        panic("load user err: " + err.Error())
    }
}

func getUser() *User {
    if user == nil {
        userOnce.Do(initUser)
    }
    return user
}

由于Golang沒有volatile關(guān)鍵字,不能控制單例在內(nèi)存的可見性,那么多goroutine并發(fā)時,就有可能出現(xiàn)這樣的執(zhí)行時序:

  • goroutine-A過了getUser的user == nil判斷,進入到了initUser邏輯,走到了cfgStr := XXX一行
  • 此時切換到goroutine-B,因為goroutine-A在initUser已經(jīng)走過了user = &User{}一行,所以跳過了user == nil判斷,直接返回沒有完全初始化的user實例,然后一直往下運行,就沒切回給goroutine-A

這樣的結(jié)果,就導(dǎo)致有g(shù)oroutine拿到未初始化完成的實例往后運行,后面就出問題了。所以實戰(zhàn)當(dāng)中需要留意,用sync.Once時,不能也不需要加這些nil判斷,就能滿足懶加載單例/配置之類的邏輯。

到此這篇關(guān)于golang使用sync.Once實現(xiàn)懶加載的用法和坑點詳解的文章就介紹到這了,更多相關(guān)go sync.Once懶加載內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • go語言工程結(jié)構(gòu)

    go語言工程結(jié)構(gòu)

    這篇文章主要簡單介紹了go語言工程結(jié)構(gòu),對于我們學(xué)習(xí)go語言很有幫助,需要的朋友可以參考下
    2015-01-01
  • 我放棄Python轉(zhuǎn)Go語言的9大理由(附優(yōu)秀書籍推薦)

    我放棄Python轉(zhuǎn)Go語言的9大理由(附優(yōu)秀書籍推薦)

    這篇文章主要給大家介紹了關(guān)于我放棄Python轉(zhuǎn)Go語言的9大理由,以及給大家推薦了6本優(yōu)秀的go語言書籍,對同樣想學(xué)習(xí)golang的朋友們具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-10-10
  • 幾個小技巧幫你實現(xiàn)Golang永久阻塞

    幾個小技巧幫你實現(xiàn)Golang永久阻塞

    Go 的運行時的當(dāng)前設(shè)計,假定程序員自己負(fù)責(zé)檢測何時終止一個 goroutine 以及何時終止該程序。有時候我們需要的是使程序阻塞在這一行,本文就來詳細(xì)的介紹一下,感興趣的可以了解一下
    2021-12-12
  • Golang實現(xiàn)不被復(fù)制的結(jié)構(gòu)體的方法

    Golang實現(xiàn)不被復(fù)制的結(jié)構(gòu)體的方法

    sync包中的許多結(jié)構(gòu)都是不允許拷貝的,因為它們自身存儲了一些狀態(tài)(比如等待者的數(shù)量),如果你嘗試復(fù)制這些結(jié)構(gòu)體,就會在你的?IDE中看到警告,那這是怎么實現(xiàn)的呢,下文就來和大家詳細(xì)講講
    2023-03-03
  • Go?結(jié)構(gòu)體序列化的實現(xiàn)

    Go?結(jié)構(gòu)體序列化的實現(xiàn)

    本文主要介紹了Go?結(jié)構(gòu)體序列化的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • 使用gRPC實現(xiàn)獲取數(shù)據(jù)庫版本

    使用gRPC實現(xiàn)獲取數(shù)據(jù)庫版本

    這篇文章主要為大家詳細(xì)介紹了如何使用gRPC實現(xiàn)獲取數(shù)據(jù)庫版本,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-12-12
  • Go 語言的指針的學(xué)習(xí)筆記

    Go 語言的指針的學(xué)習(xí)筆記

    這篇文章主要介紹了Go 語言的指針的學(xué)習(xí)筆記,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-09-09
  • golang elasticsearch Client的使用詳解

    golang elasticsearch Client的使用詳解

    這篇文章主要介紹了golang elasticsearch Client的使用詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • go語言VScode?see?'go?help?modules'?(exit?status?1)問題的解決過程

    go語言VScode?see?'go?help?modules'?(exit?statu

    最近上手學(xué)習(xí)go語言,準(zhǔn)備在VSCode上寫程序的時候卻發(fā)現(xiàn)出了一點問題,下面這篇文章主要給大家介紹了關(guān)于go語言VScode?see?'go?help?modules'(exit?status?1)問題的解決過程,需要的朋友可以參考下
    2022-07-07
  • 詳解Go語言如何使用標(biāo)準(zhǔn)庫sort對切片進行排序

    詳解Go語言如何使用標(biāo)準(zhǔn)庫sort對切片進行排序

    Sort?標(biāo)準(zhǔn)庫提供了對基本數(shù)據(jù)類型的切片和自定義類型的切片進行排序的函數(shù)。今天主要分享的內(nèi)容是使用?Go?標(biāo)準(zhǔn)庫?sort?對切片進行排序,感興趣的可以了解一下
    2022-12-12

最新評論