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

Golang Goroutine的使用

 更新時(shí)間:2020年10月21日 11:42:23   作者:李培冠  
這篇文章主要介紹了Golang Goroutine的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

什么是 Goroutine

goroutine 是 Go 并行設(shè)計(jì)的核心。goroutine 說到底其實(shí)就是協(xié)程,它比線程更小,十幾個(gè) goroutine 可能體現(xiàn)在底層就是五六個(gè)線程,Go 語言內(nèi)部幫你實(shí)現(xiàn)了這些 goroutine 之間的內(nèi)存共享。

執(zhí)行 goroutine 只需極少的棧內(nèi)存(大概是4~5KB),當(dāng)然會根據(jù)相應(yīng)的數(shù)據(jù)伸縮。也正因?yàn)槿绱?,可同時(shí)運(yùn)行成千上萬個(gè)并發(fā)任務(wù)。goroutine 比 thread 更易用、更高效、更輕便。

一般情況下,一個(gè)普通計(jì)算機(jī)跑幾十個(gè)線程就有點(diǎn)負(fù)載過大了,但是同樣的機(jī)器卻可以輕松地讓成百上千個(gè) goroutine 進(jìn)行資源競爭。

Goroutine 的創(chuàng)建

只需在函數(shù)調(diào)⽤語句前添加 go 關(guān)鍵字,就可創(chuàng)建并發(fā)執(zhí)⾏單元。

開發(fā)⼈員無需了解任何執(zhí)⾏細(xì)節(jié),調(diào)度器會自動(dòng)將其安排到合適的系統(tǒng)線程上執(zhí)行。

在并發(fā)編程中,我們通常想將一個(gè)過程切分成幾塊,然后讓每個(gè) goroutine 各自負(fù)責(zé)一塊工作,當(dāng)一個(gè)程序啟動(dòng)時(shí),主函數(shù)在一個(gè)單獨(dú)的 goroutine 中運(yùn)行,我們叫它 main goroutine。新的 goroutine 會用 go 語句來創(chuàng)建。而 go 語言的并發(fā)設(shè)計(jì),讓我們很輕松就可以達(dá)成這一目的。

例如:

package main

import (
 "fmt"
 "time"
)

func foo() {
 i := 0
 for true {
 i++
 fmt.Println("new goroutine: i = ", i)
 time.Sleep(time.Second)
 }
}

func main() {
 // 創(chuàng)建一個(gè) goroutine, 啟動(dòng)另外一個(gè)任務(wù)
 go foo()

 i := 0
 for true {
 i++
 fmt.Println("main goroutine: i = ", i)
 time.Sleep(time.Second)
 }
}

結(jié)果:

main goroutine: i =  1
new goroutine: i =  1
new goroutine: i =  2
main goroutine: i =  2
main goroutine: i =  3
new goroutine: i =  3
...

Goroutine 特性

主go程 退出后,其它的 子go程 也會自動(dòng)退出:

package main

import (
 "fmt"
 "time"
)

func foo() {
 i := 0
 for true {
 i++
 fmt.Println("new goroutine: i = ", i)
 time.Sleep(time.Second)
 }
}

func main() {
 // 創(chuàng)建一個(gè) goroutine, 啟動(dòng)另外一個(gè)任務(wù)
 go foo()

 time.Sleep(time.Second * 3)

 fmt.Println("main goroutine exit")
}

運(yùn)行結(jié)果:

new goroutine: i =  1
new goroutine: i =  2
new goroutine: i =  3
main goroutine exit

runtime 包

Gosched

runtime.Gosched() 用于出讓當(dāng)前 go程 所占用的 CPU 時(shí)間片,讓出當(dāng)前 goroutine 的執(zhí)行權(quán)限,調(diào)度器安排其他等待的任務(wù)運(yùn)行,并在下次再獲得 cpu 時(shí)間輪片的時(shí)候,從該出讓 cpu 的位置恢復(fù)執(zhí)行。

有點(diǎn)像跑接力賽,A 跑了一會碰到代碼 runtime.Gosched() 就把接力棒交給 B 了,A 歇著了,B 繼續(xù)跑。

例如:

package main

import (
 "fmt"
 "runtime"
 "time"
)

func main() {
 // 創(chuàng)建一個(gè) goroutine
 go func(s string) {
 for i := 0; i < 2; i++ {
  fmt.Println(s)
 }
 }("world")

 for i := 0; i < 2; i++ {
 runtime.Gosched()
 fmt.Println("hello")
 }
 time.Sleep(time.Second * 3)
}

運(yùn)行結(jié)果:

world
world
hello
hello

如果沒有 runtime.Gosched() 則運(yùn)行結(jié)果如下:

hello
hello
world
world

注意: runtime.Gosched() 只是出讓一次機(jī)會,看下面的代碼,注意運(yùn)行結(jié)果:

package main

import (
 "fmt"
 "runtime"
 "time"
)

func main() {
 // 創(chuàng)建一個(gè) goroutine
 go func(s string) {
 for i := 0; i < 2; i++ {
  fmt.Println(s)
  time.Sleep(time.Second)
 }
 }("world")

 for i := 0; i < 2; i++ {
 runtime.Gosched()
 fmt.Println("hello")
 }
}

運(yùn)行結(jié)果:

world
hello
hello

為什么 world 只有一次呢?因?yàn)橹拔覀冋f過,主 goroutine 退出后,其它的工作 goroutine 也會自動(dòng)退出。

Goexit

調(diào)用 runtime.Goexit() 將立即終止當(dāng)前 goroutine 執(zhí)⾏,調(diào)度器確保所有已注冊 defer 延遲調(diào)用被執(zhí)行。

注意與 return 的區(qū)別,return 是返回當(dāng)前函數(shù)調(diào)用給調(diào)用者。

例如:

package main

import (
 "fmt"
 "runtime"
 "time"
)

func main() {
 go func() {
 defer fmt.Println("A.defer")
 func() {
  defer fmt.Println("B.defer")
  runtime.Goexit() // 終止當(dāng)前 goroutine
  fmt.Println("B") // 不會執(zhí)行
 }()
 fmt.Println("A") // 不會執(zhí)行
 }() // 不要忘記 ()

 time.Sleep(time.Second * 3)
}

運(yùn)行結(jié)果:

B.defer
A.defer

GOMAXPROCS

調(diào)用 runtime.GOMAXPROCS() 用來設(shè)置可以并行計(jì)算的 CPU 核數(shù)的最大值,并返回 上一次(沒有則是電腦默認(rèn)的) 設(shè)置的值。

package main

import (
 "fmt"
 "runtime"
)

func main() {
 runtime.GOMAXPROCS(1) // 將 cpu 設(shè)置為單核

 for true {
 go fmt.Print(0) // 子 go 程
 fmt.Print(1) // 主 go 程
 }
}

運(yùn)行結(jié)果:

111111 ... 1000000 ... 0111 ...

在執(zhí)行 runtime.GOMAXPROCS(1) 時(shí),最多同時(shí)只能有一個(gè) goroutine 被執(zhí)行。所以會打印很多 1。過了一段時(shí)間后,GO 調(diào)度器會將其置為休眠,并喚醒另一個(gè) goroutine,這時(shí)候就開始打印很多 0 了,在打印的時(shí)候,goroutine 是被調(diào)度到操作系統(tǒng)線程上的。

package main

import (
 "fmt"
 "runtime"
)

func main() {
 runtime.GOMAXPROCS(2)

 for true {
 go fmt.Print(0)
 fmt.Print(1)
 }
}

運(yùn)行結(jié)果:

111111111111111000000000000000111111111111111110000000000000000011111111100000...

在執(zhí)行 runtime.GOMAXPROCS(2) 時(shí), 我們使用了兩個(gè) CPU,所以兩個(gè) goroutine 可以一起被執(zhí)行,以同樣的頻率交替打印 0 和 1。

runtime 包中的其它函數(shù)

中文文檔在這里:https://studygolang.com/pkgdoc

這里就簡單列舉一下一些函數(shù)以及功能。

func GOROOT() string

GOROOT 返回 Go 的根目錄。如果存在 GOROOT 環(huán)境變量,返回該變量的值;否則,返回創(chuàng)建 Go 時(shí)的根目錄。

func Version() string

返回 Go 的版本字符串。它要么是遞交的 hash 和創(chuàng)建時(shí)的日期;要么是發(fā)行標(biāo)簽如 "go1.3"。

func NumCPU() int

NumCPU返回本地機(jī)器的邏輯CPU個(gè)數(shù)(真 · 八核)。

func GC()

GC執(zhí)行一次垃圾回收。(如果你迫切的希望做一次垃圾回收,可以調(diào)用此函數(shù))

到此這篇關(guān)于Golang Goroutine的使用的文章就介紹到這了,更多相關(guān)Golang Goroutine內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go語言常見錯(cuò)誤之誤用init函數(shù)實(shí)例解析

    Go語言常見錯(cuò)誤之誤用init函數(shù)實(shí)例解析

    Go語言中的init函數(shù)為開發(fā)者提供了一種在程序正式運(yùn)行前初始化包級變量的機(jī)制,然而,由于init函數(shù)的特殊性,不當(dāng)?shù)厥褂盟赡芤鹨幌盗袉栴},本文將深入探討如何有效地使用init函數(shù),列舉常見誤用并提供相應(yīng)的避免策略
    2024-01-01
  • Go語言為什么很少使用數(shù)組原理解析

    Go語言為什么很少使用數(shù)組原理解析

    這篇文章主要為大家介紹了Go語言為什么很少使用數(shù)組原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12
  • golang中json反序列化可能遇到的問題

    golang中json反序列化可能遇到的問題

    這篇文章主要給大家介紹了關(guān)于golang中json反序列化可能遇到的問題的解決方法,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2018-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í)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-10-10
  • Golang實(shí)現(xiàn)深拷貝reflect原理示例探究

    Golang實(shí)現(xiàn)深拷貝reflect原理示例探究

    這篇文章主要為大家介紹了Golang實(shí)現(xiàn)reflect深拷貝原理示例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • golang import自定義包方式

    golang import自定義包方式

    這篇文章主要介紹了golang import自定義包方式,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • 如何用golang運(yùn)行第一個(gè)項(xiàng)目

    如何用golang運(yùn)行第一個(gè)項(xiàng)目

    這篇文章主要介紹了如何用golang運(yùn)行第一個(gè)項(xiàng)目,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-03-03
  • 淺談Golang的new與make區(qū)別是什么

    淺談Golang的new與make區(qū)別是什么

    本文主要介紹了Golang的new與make區(qū)別是什么,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • Go類型斷言提取測試接口值動(dòng)態(tài)類型及靜態(tài)轉(zhuǎn)換確保類型接口編譯安全

    Go類型斷言提取測試接口值動(dòng)態(tài)類型及靜態(tài)轉(zhuǎn)換確保類型接口編譯安全

    這篇文章主要為大家介紹了Go類型斷言提取測試接口值動(dòng)態(tài)類型及靜態(tài)轉(zhuǎn)換確保類型實(shí)現(xiàn)特定接口的編譯時(shí)安全性詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • Go結(jié)構(gòu)體指針引發(fā)的值傳遞思考分析

    Go結(jié)構(gòu)體指針引發(fā)的值傳遞思考分析

    這篇文章主要為大家介紹了Go結(jié)構(gòu)體指針引發(fā)的值傳遞思考分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12

最新評論