日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無(wú)先后,達(dá)者為師

網(wǎng)站首頁(yè) 編程語(yǔ)言 正文

Golang優(yōu)雅保持main函數(shù)不退出的辦法_Golang

作者:MiKogy_ ? 更新時(shí)間: 2022-09-13 編程語(yǔ)言

高能預(yù)警

本文包含演示部分,請(qǐng)讀者自行copy代碼編譯體驗(yàn)。

參考資料:sync.WaitGroup / signal.Notify / context.CancelFunc

正文

我們有時(shí)會(huì)希望我們的程序保持執(zhí)行,但是有一種情況是:我們的代碼全部塞入go routine時(shí),主函數(shù)會(huì)立刻退出,本文將和大家分享如何讓main函數(shù)優(yōu)雅地保持執(zhí)行。

問(wèn)題演示:

func main() {
    go func() {
        for i := 0; i<10000;i ++ {
            fmt.Println(i)
        }
    }()
}

此時(shí)我們可以看到,控制臺(tái)幾乎不會(huì)輸出任何內(nèi)容。究其原因,是主函數(shù)在go routine執(zhí)行前就已經(jīng)結(jié)束,也就是說(shuō)go routine不會(huì)阻塞主函數(shù)。

可能有些讀者會(huì)想到,我直接加個(gè)死循環(huán)在下面,讓主函數(shù)不退出不就行啦?博主表示十分贊同,因?yàn)椴┲骶褪遣捎眠@個(gè)方法,導(dǎo)致服務(wù)器跑滿CPU從而不停的告警。

那么解決辦法是:讓死循環(huán)慢一點(diǎn)執(zhí)行,即添加以下內(nèi)容:

    for {
        time.Sleep(time.Second)
    }

但是在博主的完美主義光環(huán)加持下,還是希望我們的代碼能更加優(yōu)雅,下面將介紹另外三種比較優(yōu)雅的保持main函數(shù)的辦法。

解決辦法演示

操作系統(tǒng)信號(hào)阻塞

先上代碼:

func main() {
    c := make(chan os.Signal)
    signal.Notify(c)
    go func() {
        fmt.Println("Go routine running")
        time.Sleep(3*time.Second)
        fmt.Println("Go routine done")
    }()
    <-c
    fmt.Println("bye")
}

官網(wǎng)機(jī)翻:signal.Notify()方法使信號(hào)將傳入c。如果沒(méi)有提供信號(hào),所有傳入的信號(hào)將被中繼到c。

  • 這里我們創(chuàng)建了一個(gè)os.Signal類型的管道。當(dāng)管道為空的時(shí)候,讀管道操作“<-”會(huì)阻塞住,直到我們向進(jìn)程發(fā)送一個(gè)信號(hào)(例如 Ctrl+C),才會(huì)繼續(xù)執(zhí)行該操作后面的代碼。

上下文操作阻塞

再上代碼:

func main() {
    ctx, cancel := context.WithCancel(context.Background())
    defer cancel()
    go func() {
        fmt.Println("Go routine running")
        time.Sleep(3 * time.Second)
        fmt.Println("Go routine done")
        cancel()
    }()
    <-ctx.Done()
    fmt.Println("bye")
}

官網(wǎng)機(jī)翻:CancelFunc() 通知操作放棄其(當(dāng)前的)工作。CancelFunc() 不會(huì)等待工作停止。

  • 這也是一個(gè)十分優(yōu)雅的辦法,我們創(chuàng)建一個(gè)可以終止的上下文——context.WithCancel(),并在go routine執(zhí)行完畢時(shí)調(diào)用其返回的CancelFunc() 方法,即表示該上下文已經(jīng)結(jié)束了。而在這之前,我們會(huì)使用<-ctx.Done()來(lái)一直等待上下文的結(jié)束,也就是說(shuō)main函數(shù)被成功阻塞,并等待go routine執(zhí)行完畢并執(zhí)行了cancel()方法后優(yōu)雅退出。

WaitGroup阻塞

然后上代碼:

func main() {
    wg := &sync.WaitGroup{}
    wg.Add(2)
    go func() {
        time.Sleep(3*time.Second)
        fmt.Println("3 second passed")
        wg.Done()
    }()
    go func() {
        time.Sleep(5*time.Second)
        fmt.Println("5 second passed")
        wg.Done()
    }()
    wg.Wait()
    fmt.Println("bye")
}

官網(wǎng)機(jī)翻:WaitGroup 等待一組 go routine 完成。主 go routine 調(diào)用 Add() 來(lái)設(shè)置要等待的 go routine 的數(shù)量。

  • 我們首先創(chuàng)建一個(gè)WaitGroup{}對(duì)象,然后調(diào)用Add()方法,在里面?zhèn)魅胛覀兘酉聛?lái)會(huì)創(chuàng)建的go routine的數(shù)量,每當(dāng)我們執(zhí)行完一個(gè)go routine時(shí),調(diào)用一次Done()方法,使得正執(zhí)行的go routine的數(shù)量減一,當(dāng)減到0時(shí),Wait()方法將不再等待(阻塞),使main函數(shù)繼續(xù)向下執(zhí)行。

小結(jié)

以上就是我們告別for {}或者select {},并優(yōu)雅地阻塞主函數(shù)的三種辦法,也是博主作為新手時(shí)對(duì)Go語(yǔ)言特性的入門級(jí)體驗(yàn)。

總結(jié)

原文鏈接:https://blog.csdn.net/Liing0/article/details/122481451

欄目分類
最近更新