Go語言的Channel遍歷方法詳解
先來看看基本的定義:
channel是Go語言中的一個核心類型,可以把它看成管道。并發(fā)核心單元通過它就可以發(fā)送或者接收數(shù)據(jù)進行通訊,這在一定程度上又進一步降低了編程的難度。
channel是一個數(shù)據(jù)類型,主要用來解決go程的同步問題以及協(xié)程之間數(shù)據(jù)共享(數(shù)據(jù)傳遞)的問題。
(1)channle 本質(zhì)上是一個數(shù)據(jù)結(jié)構(gòu)——(隊列),數(shù)據(jù)是先進先出。
(2)具有線程安全機制,多個go程訪問時,不需要枷鎖,也就是說channel本身是線程安全的。
(3)channel是有類型的,如一個string類型的channel只能存放string類型數(shù)據(jù)。
Channel遍歷主要分為3種:
1)簡單的讀 data:=<-ch (如果讀多次,需要用循環(huán))
var ch8 = make(chan int, 6)
func mm1() {
for i := 0; i < 10; i++ {
ch8 <- 8 * i
}
}
func main() {
go mm1()
for i:=0;i<100;i++{
fmt.Print(<-ch8, "\t")
}
}

注:
(1)寫入的次數(shù)與讀取的次數(shù)需要一致(本例是10);
(2)如果讀的次數(shù)多于寫的次數(shù)會發(fā)生:fatal error: all goroutines are asleep - deadlock! ,若 在mm1中對ch8進行關(guān)閉(執(zhí)行 close(ch8) ),多于的次數(shù)讀到的數(shù)據(jù)為0(數(shù)據(jù)默認值)。
(3)讀的次數(shù)少于寫的次數(shù),會讀取出次數(shù)對應的內(nèi)容,不會報錯。
2)斷言方式
if value, ok := <-ch; ok == true {
1) 如果寫端沒有寫數(shù)據(jù),也沒有關(guān)閉。<-ch; 會阻塞 ---【重點】
2)如果寫端寫數(shù)據(jù), value 保存 <-ch 讀到的數(shù)據(jù)。 ok 被設(shè)置為 true
3)如果寫端關(guān)閉。 value 為數(shù)據(jù)類型默認值。ok 被設(shè)置為 false
var ch8 = make(chan int, 6)
func mm1() {
for i := 0; i < 10; i++ {
ch8 <- 8 * i
}
close(ch8)
}
func main() {
go mm1()
for {
if data, ok := <-ch8; ok {
fmt.Print(data,"\t")
} else {
break
}
}
}

注:寫完之后一定要關(guān)閉( 執(zhí)行:close(ch8) ),否則會出現(xiàn)以下運行結(jié)果:

3)通過range方法
for num := range ch {
}
var ch8 = make(chan int, 6)
func mm1() {
for i := 0; i < 10; i++ {
ch8 <- 8 * i
}
close(ch8)
}
func main() {
go mm1()
for {
for data := range ch8 {
fmt.Print(data,"\t")
}
break
}
}
注:寫完之后一定要關(guān)閉( 執(zhí)行:close(ch8) ),否則會出現(xiàn)以下運行結(jié)果:

特別說明:以上實例都是子go程寫,主go程讀。如在子go程中寫,另一個子go程中讀,不管哪種方法,都不會出現(xiàn)以上錯誤問題。(多次實例驗證)
var ch8 = make(chan int, 6)
func mm1() {
for i := 0; i < 10; i++ {
ch8 <- 8 * i
}
//close(ch8)
}
func mm2() {
for {
for data:=range ch8{
fmt.Print(data,"\t")
}
}
}
func main() {
go mm1()
go mm2()
for{
runtime.GC()
}
}

總結(jié):
通過以上驗證,為了保證程序的健壯性,在設(shè)計程序時,最好將channel的讀、寫分別在子go程中進行。寫完數(shù)據(jù)之后,記得關(guān)閉channel。
補充一點:
1、channel不像文件一樣需要經(jīng)常去關(guān)閉,只有當你確實沒有任何發(fā)送數(shù)據(jù)了,或者你想顯式的結(jié)束range循環(huán)之類的,才去關(guān)閉channel;
2、關(guān)閉channel后,無法向channel 再發(fā)送數(shù)據(jù)(引發(fā) panic 錯誤后導致接收立即返回零值);
3、關(guān)閉channel后,可以繼續(xù)從channel接收數(shù)據(jù);
4、對于nil channel,無論收發(fā)都會被阻塞。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
Golang?WebSocket創(chuàng)建單獨會話詳細實例
這篇文章主要給大家介紹了關(guān)于Golang?WebSocket創(chuàng)建單獨會話的相關(guān)資料,WebSocket 協(xié)議主要為了解決基于 HTTP/1.x 的 Web 應用無法實現(xiàn)服務端向客戶端主動推送的問題,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2023-11-11
GoFrame?gmap遍歷hashmap?listmap?treemap使用技巧
這篇文章主要為大家介紹了GoFrame?gmap遍歷hashmap?listmap?treemap使用技巧的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06
Golang程序中使用Prometheus的client_golang庫
這篇文章主要介紹了Golang程序中使用Prometheus的client_golang庫,Prometheus 是一個開源的監(jiān)控和警報工具包,用于收集和處理應用程序和系統(tǒng)的指標數(shù)據(jù)。Prometheus 提供了多種客戶端庫,可以輕松地集成到各種編程語言中2023-04-04

