一圖勝千言,幫你搞懂Go面試中常問的channel問題!
一圖勝千言
下面的表格中總結(jié)了對(duì)不同狀態(tài)下的通道執(zhí)行相應(yīng)操作的結(jié)果。
注意:對(duì)已經(jīng)關(guān)閉的通道再執(zhí)行 close 也會(huì)引發(fā) panic。
這篇文章將重點(diǎn)講解Go面試進(jìn)階知識(shí)點(diǎn):select和channel。
select
先說(shuō)switch...case...
switch...case... 很常用,且很好理解。其作用和if...else...一樣。
區(qū)別是switch...case 相比于if...else...能讓我們的代碼看起來(lái)更清晰,更好理解。
再說(shuō)select...case..
golang 的 select 就是監(jiān)聽 IO 操作,當(dāng) IO 操作發(fā)生時(shí),觸發(fā)相應(yīng)的動(dòng)作。
所說(shuō)的IO操作就是對(duì)channle的操作:向通道發(fā)送數(shù)據(jù),或者從通道中讀取數(shù)據(jù)。
在執(zhí)行select語(yǔ)句的時(shí)候,運(yùn)行時(shí)系統(tǒng)會(huì)自上而下地判斷每個(gè)case中的發(fā)送或接收操作是否可以被立即執(zhí)行。
什么是立即執(zhí)行呢?
立即執(zhí)行:意思是當(dāng)前Goroutine不會(huì)因當(dāng)前操作而被阻塞
select類比switch
select的用法與switch非常類似,由select開始一個(gè)新的選擇塊,每個(gè)選擇條件由case語(yǔ)句來(lái)描述。
與switch語(yǔ)句可以選擇任何可使用相等比較的條件相比,select有比較多的限制,其中最大的一條限制就是每個(gè)case語(yǔ)句里必須是一個(gè)IO操作。
確切的說(shuō),應(yīng)該是一個(gè)面向channel的IO操作。
經(jīng)典示例
package main
import "fmt"
func main() {
ch1 := make(chan int, 1)
ch1 <- 2
select {
case v := <-ch1:
fmt.Println("取到的數(shù)據(jù):", v)
case ch1 <- 1:
fmt.Println("寫入數(shù)據(jù)")
}
}
運(yùn)行結(jié)果
channel
goroutine和channel作為go語(yǔ)言中最重要的兩個(gè)知識(shí)點(diǎn),一定要搞清楚。
大家容易出錯(cuò)的知識(shí)點(diǎn)是以下3點(diǎn),尤其是最后一點(diǎn):
- nil channel代表channel未初始化,向未初始化的channel讀寫數(shù)據(jù)會(huì)造成阻塞
- 關(guān)閉(close)未初始化的channel會(huì)引起panic。
- 從一個(gè)關(guān)閉的并且沒有值的通道執(zhí)行接收操作會(huì)得到對(duì)應(yīng)類型的零值,并不會(huì)引起panic。
1.從已經(jīng)關(guān)閉并且沒有值的通道中取值
package main
import "fmt"
//從關(guān)閉的通道中取值示例:
func main() {
//聲明實(shí)例化通道ch1
ch1 := make(chan int, 1)
//關(guān)閉通道
close(ch1)
select {
//通通道ch1中取值
case v := <-ch1:
fmt.Printf("從ch1中取值:%d\n", v)
default:
fmt.Println("默認(rèn)case")
}
}
運(yùn)行結(jié)果
和我們預(yù)想中的一樣,取到了對(duì)應(yīng)的零值:
2.從已經(jīng)關(guān)閉并且有值的通道中取值
我們稍微修改一下上面的代碼
package main
import "fmt"
//從關(guān)閉的通道中取值示例:
func main() {
//聲明實(shí)例化通道ch1
ch1 := make(chan int, 1)
//向通道中賦值
ch1 <- 1
//關(guān)閉通道
close(ch1)
//關(guān)閉之后取值
after_close_value := <-ch1
fmt.Printf("關(guān)閉之后取值:%d\n", after_close_value) //打印結(jié)果:關(guān)閉之后取值:1
select {
//通通道ch1中取值
case v := <-ch1:
fmt.Printf("從ch1中取值:%d\n", v) //打印結(jié)果:從ch1中取值:0
default:
fmt.Println("默認(rèn)case")
}
}
運(yùn)行結(jié)果
運(yùn)行結(jié)果和我們預(yù)想中的一樣:
- 通道關(guān)閉后,如果通道中仍然有值,還是可以正常取到通道中的值的。
- 通道關(guān)閉后,如果通道中已經(jīng)沒有值了,再?gòu)耐ǖ乐腥≈?,并不?huì)引起panic,而是會(huì)取到對(duì)應(yīng)類型的零值。
一圖勝千言
下面的表格中總結(jié)了對(duì)不同狀態(tài)下的通道執(zhí)行相應(yīng)操作的結(jié)果。
注意:對(duì)已經(jīng)關(guān)閉的通道再執(zhí)行 close 也會(huì)引發(fā) panic。
總結(jié)
這篇文章解析了Go語(yǔ)言中select和channel在面試中可能遇到的進(jìn)階知識(shí)點(diǎn)。
本文轉(zhuǎn)載自微信公眾號(hào)「 程序員升級(jí)打怪之旅」,作者「王中陽(yáng)Go」,可以通過以下二維碼關(guān)注。
轉(zhuǎn)載本文請(qǐng)聯(lián)系「 程序員升級(jí)打怪之旅」公眾號(hào)。