深入淺出:讓你真正理解Golang并發(fā)模型
在當(dāng)今互聯(lián)網(wǎng)領(lǐng)域,Golang這門語(yǔ)言被越來(lái)越多的技術(shù)從業(yè)者所熟知和使用。Golang作為一門高效、簡(jiǎn)潔、并發(fā)性強(qiáng)的語(yǔ)言,其并發(fā)模型在眾多編程語(yǔ)言中也很獨(dú)特。那么,在本文中,我們將深入淺出的講解Golang并發(fā)模型的相關(guān)知識(shí)點(diǎn),為大家剖析Golang并發(fā)模型的精髓。
1. Golang Goroutine和Channel
Goroutine和Channel是Golang并發(fā)模型的兩個(gè)重要組成部分。Goroutine是一個(gè)輕量級(jí)的線程,可以讓程序同時(shí)執(zhí)行多個(gè)任務(wù),而Channel則是多個(gè)Goroutine之間進(jìn)行通信的重要機(jī)制。
Goroutine的啟動(dòng)非常簡(jiǎn)單,只需要在函數(shù)前面加上關(guān)鍵字go即可。例如:
`go
func main() {
go foo()
go bar()
}
func foo() {
// 這里是foo的任務(wù)邏輯
}
func bar() {
// 這里是bar的任務(wù)邏輯
}
在上面的代碼中,我們可以看到主函數(shù)中啟動(dòng)了兩個(gè)Goroutine,分別是foo和bar。這樣,我們就可以同時(shí)執(zhí)行多個(gè)任務(wù)了。Channel則是用于實(shí)現(xiàn)Goroutine之間通信的重要機(jī)制。通常來(lái)說,一個(gè)Goroutine在執(zhí)行任務(wù)的時(shí)候,可能需要取得另一個(gè)Goroutine的執(zhí)行結(jié)果。這時(shí)候,我們就可以使用Channel來(lái)進(jìn)行通信。`gofunc main() { c := make(chan int) go sum(1, 2, c) result := <-c fmt.Println(result)}func sum(a int, b int, c chan int) { sum := a + b c <- sum}
在上面的代碼中,我們定義了一個(gè)Channel,然后啟動(dòng)了一個(gè)Goroutine來(lái)執(zhí)行sum函數(shù),將執(zhí)行結(jié)果通過Channel發(fā)送出去。在主函數(shù)中,我們則通過<-c的方式來(lái)接受Channel中的值。這樣,我們就可以實(shí)現(xiàn)Goroutine之間的通信了。
2. Golang的Mutex和WaitGroup
除了Goroutine和Channel之外,Golang并發(fā)模型中還有兩個(gè)非常重要的概念,分別是Mutex和WaitGroup。
Mutex是Golang中的互斥鎖,用于在多個(gè)Goroutine之間保持排他性。如果多個(gè)Goroutine同時(shí)對(duì)某個(gè)共享變量進(jìn)行讀寫,會(huì)導(dǎo)致數(shù)據(jù)錯(cuò)亂,使用Mutex可以解決這個(gè)問題。
`go
var mutex sync.Mutex
func main() {
go foo()
go bar()
}
func foo() {
mutex.Lock()
// 這里是foo的任務(wù)邏輯
mutex.Unlock()
}
func bar() {
mutex.Lock()
// 這里是bar的任務(wù)邏輯
mutex.Unlock()
}
在上面的代碼中,我們定義了一個(gè)全局的互斥鎖mutex,然后在foo和bar函數(shù)中,對(duì)共享變量進(jìn)行了讀寫。在對(duì)共享變量進(jìn)行讀寫的時(shí)候,我們使用了mutex.Lock()和mutex.Unlock()來(lái)保持排他性。WaitGroup則是Golang中的等待組,用于等待多個(gè)Goroutine任務(wù)的完成。如果我們希望主程序等待多個(gè)Goroutine任務(wù)執(zhí)行完成之后再結(jié)束,可以使用WaitGroup來(lái)實(shí)現(xiàn)。`govar wg sync.WaitGroupfunc main() { wg.Add(2) go foo() go bar() wg.Wait() fmt.Println("任務(wù)執(zhí)行完成")}func foo() { defer wg.Done() // 這里是foo的任務(wù)邏輯}func bar() { defer wg.Done() // 這里是bar的任務(wù)邏輯}
在上面的代碼中,我們定義了一個(gè)全局的等待組wg,并且在主函數(shù)中調(diào)用了wg.Wait()進(jìn)行等待。在foo和bar函數(shù)中,我們則分別調(diào)用了wg.Done()來(lái)通知等待組任務(wù)已經(jīng)完成。這樣,當(dāng)兩個(gè)任務(wù)執(zhí)行完成之后,主程序就會(huì)退出了。
3. Golang Select和Timeout
Golang中的Select和Timeout也是Golang并發(fā)模型中的兩個(gè)非常重要的概念。Select可以用來(lái)在多個(gè)Channel中選擇執(zhí)行操作,而Timeout則可以用來(lái)設(shè)置操作超時(shí)時(shí)間。
`go
func main() {
c1 := make(chan int)
c2 := make(chan int)
go func() {
time.Sleep(time.Second)
c1 <- 1
}()
go func() {
time.Sleep(time.Second * 2)
c2 <- 2
}()
select {
case r1 := <-c1:
fmt.Println("收到c1結(jié)果:", r1)
case r2 := <-c2:
fmt.Println("收到c2結(jié)果:", r2)
case <-time.After(time.Second * 3):
fmt.Println("執(zhí)行超時(shí)")
}
}
在上面的代碼中,我們定義了兩個(gè)Channel,然后啟動(dòng)了兩個(gè)Goroutine分別向兩個(gè)Channel中發(fā)送數(shù)據(jù)。在主函數(shù)中,我們則使用select來(lái)選擇執(zhí)行操作,如果在三秒內(nèi)收到其中一個(gè)Channel的結(jié)果,則打印出結(jié)果,否則打印出執(zhí)行超時(shí)。另外,在使用Channel的時(shí)候,我們還可以通過向Channel中添加第二個(gè)返回值,來(lái)判斷是否成功發(fā)送或接收數(shù)據(jù)。`gofunc main() { c := make(chan int) go func() { ret, ok := <-c fmt.Println(ret, ok) }() c <- 1 close(c)}
在上面的代碼中,我們啟動(dòng)一個(gè)Goroutine來(lái)從Channel中接收數(shù)據(jù),并打印出其成功接收的結(jié)果。在主函數(shù)中,我們則向Channel中發(fā)送數(shù)據(jù),并通過close函數(shù)關(guān)閉Channel。這樣,當(dāng)Goroutine從Channel中讀取到關(guān)閉通知的時(shí)候,就可以結(jié)束執(zhí)行了。
總結(jié)
在本文中,我們?yōu)榇蠹抑v解了Golang并發(fā)模型的相關(guān)知識(shí)點(diǎn),包括Goroutine、Channel、Mutex、WaitGroup、Select和Timeout等。這些知識(shí)點(diǎn)是Golang并發(fā)模型的基礎(chǔ),如果大家能夠深入理解并掌握,就可以寫出高效、優(yōu)美并發(fā)程序。
以上就是IT培訓(xùn)機(jī)構(gòu)千鋒教育提供的相關(guān)內(nèi)容,如果您有web前端培訓(xùn),鴻蒙開發(fā)培訓(xùn),python培訓(xùn),linux培訓(xùn),java培訓(xùn),UI設(shè)計(jì)培訓(xùn)等需求,歡迎隨時(shí)聯(lián)系千鋒教育。