AVt天堂网 手机版,亚洲va久久久噜噜噜久久4399,天天综合亚洲色在线精品,亚洲一级Av无码毛片久久精品

當前位置:首頁 > 科技  > 軟件

聊聊協(xié)程和管道—管道

來源: 責編: 時間:2023-08-09 23:03:10 335觀看
導讀管道簡介【1】管道(channel)特質(zhì)介紹:(1)管道本質(zhì)就是一個數(shù)據(jù)結(jié)構(gòu)-隊列(2)數(shù)據(jù)是先進先出(3)自身線程安全,多協(xié)程訪問時,不需要加鎖,channel本身就是線程安全的(4)管道有類型的,一個string的管道只能存放string類型數(shù)據(jù)管道入門案例

管道簡介

【1】管道(channel)特質(zhì)介紹:

(1)管道本質(zhì)就是一個數(shù)據(jù)結(jié)構(gòu)-隊列Ibr28資訊網(wǎng)——每日最新資訊28at.com

(2)數(shù)據(jù)是先進先出Ibr28資訊網(wǎng)——每日最新資訊28at.com

(3)自身線程安全,多協(xié)程訪問時,不需要加鎖,channel本身就是線程安全的Ibr28資訊網(wǎng)——每日最新資訊28at.com

(4)管道有類型的,一個string的管道只能存放string類型數(shù)據(jù)Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

管道入門案例

【1】管道的定義:

var 變量名 chan 數(shù)據(jù)類型Ibr28資訊網(wǎng)——每日最新資訊28at.com

PS1:chan管道關(guān)鍵字Ibr28資訊網(wǎng)——每日最新資訊28at.com

PS2:數(shù)據(jù)類型指的是管道的類型,里面放入數(shù)據(jù)的類型,管道是有類型的,int類型的管道只能寫入整數(shù)intIbr28資訊網(wǎng)——每日最新資訊28at.com

PS3:管道是引用類型,必須初始化才能寫入數(shù)據(jù),即make后才能使用Ibr28資訊網(wǎng)——每日最新資訊28at.com

【2】案例:

func main()  {	//定義管道 、 聲明管道 ---> 定義一個int類型的管道	var intChan chan int	//通過make初始化:管道可以存放3個int類型的數(shù)據(jù)	intChan = make(chan int, 3)	//證明管道是引用類型:	fmt.Printf("intChan的值: %v /n",intChan)	//向管道存放數(shù)據(jù):	intChan <- -10	num := 20	intChan <- num	intChan <- 40	//注意:不能存放大于容量的數(shù)據(jù):	// intChan <- -80	//輸出管道的長度:	fmt.Printf("管道的實際長度:%v,管道的容量是:%v /n",len(intChan),cap(intChan))	//在管道中讀取數(shù)據(jù):	num1 := <-intChan	num2 := <-intChan	num3 := <-intChan	fmt.Println(num1)	fmt.Println(num2)	fmt.Println(num3)	//注意:在沒有使用協(xié)程的情況下,如果管道的數(shù)據(jù)已經(jīng)全部取出,那么再取就會報錯:	// num4 := <-intChan	// fmt.Println(num4)	fmt.Printf("管道的實際長度:%v,管道的容量是:%v /n",len(intChan),cap(intChan))}

管道的關(guān)閉

【1】管道的關(guān)閉:

使用內(nèi)置函數(shù)close可以關(guān)閉管道,當管道關(guān)閉后,就不能再向管道寫數(shù)據(jù)了,但是仍然可以從該管道讀取數(shù)據(jù)。Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

【2】案例:

func main()  {	var intChan chan int	intChan = make(chan int, 3)	intChan <- 10	intChan <- 20	//關(guān)閉管道:	close(intChan)	//再次寫入數(shù)據(jù):--->報錯	// intChan <- 30	//當管道關(guān)閉后,讀取數(shù)據(jù)是可以的:	num := <- intChan	fmt.Println(num)}

管道的遍歷

【1】管道的遍歷:

管道支持for-range的方式進行遍歷,請注意兩個細節(jié)Ibr28資訊網(wǎng)——每日最新資訊28at.com

1)在遍歷時,如果管道沒有關(guān)閉,則會出現(xiàn)deadlock的錯誤Ibr28資訊網(wǎng)——每日最新資訊28at.com

2)在遍歷時,如果管道已經(jīng)關(guān)閉,則會正常遍歷數(shù)據(jù),遍歷完后,就會退出遍歷。Ibr28資訊網(wǎng)——每日最新資訊28at.com

【2】案例:

func main()  {	var intChan chan int	intChan = make(chan int, 100)	for i := 0; i < 100; i++ {		intChan <- i	}	//在遍歷前,如果沒有關(guān)閉管道,就會出現(xiàn)deadlock的錯誤	//所以我們在遍歷前要進行管道的關(guān)閉	// for v := range intChan {	// 	fmt.Println("value = ",v)	// }	close(intChan)	//遍歷:for-range	for v := range intChan {		fmt.Println("value = ",v)	}}

協(xié)程和管道協(xié)同工作案例

【1】案例需求:

請完成協(xié)程和管道協(xié)同工作的案例,具體要求:Ibr28資訊網(wǎng)——每日最新資訊28at.com

1) 開啟一個writeData協(xié)程,向管道中寫入50個整數(shù).Ibr28資訊網(wǎng)——每日最新資訊28at.com

2) 開啟一個readData協(xié)程,從管道中讀取writeData寫入的數(shù)據(jù)。Ibr28資訊網(wǎng)——每日最新資訊28at.com

3) 注意: writeData和readDate操作的是同一個管道Ibr28資訊網(wǎng)——每日最新資訊28at.com

4) 主線程需要等待writeData和readDate協(xié)程都完成工作才能退出Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

【2】原理圖:

Ibr28資訊網(wǎng)——每日最新資訊28at.com

package mainimport (	"fmt"	"time"	"sync")var wg sync.WaitGroup//寫:func writeData(intChan chan int)  {	defer wg.Done()	for i := 1; i <= 50; i++ {		intChan <- i		fmt.Println("寫入的數(shù)據(jù)為:",i)		time.Sleep(time.Second)	}		close(intChan)}//讀:func readData(intChan chan int) {	defer wg.Done()	for v := range intChan {		fmt.Println("讀取的數(shù)據(jù)為:",v)		time.Sleep(time.Second)	}}func main()  {	//主線程	//寫協(xié)程和讀協(xié)程共同操作同一個管道-》定義管道:	intChan := make(chan int, 50)	wg.Add(2)	//開啟讀和寫的協(xié)程:	go writeData(intChan)	go readData(intChan)	//主線程一直在阻塞,什么時候wg減為0了,就停止	wg.Wait()	fmt.Println("讀寫數(shù)據(jù)完成...")}

運行結(jié)果:Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

聲明只讀只寫管道

【1】管道可以聲明為只讀或者只寫性質(zhì)

【2】代碼:

package mainimport (	"fmt")func main()  {	//默認情況下,管道是雙向的--》可讀可寫:	//聲明為只寫:	// 管道具備<- 只寫性質(zhì)	var intChan chan<- int	intChan = make(chan int, 3)	intChan <- 10	// 報錯	// num := <- intChan	fmt.Println("intChan:",intChan)	//聲明為只讀:	// 管道具備<- 只讀性質(zhì) 	var intChan2 <-chan int	if intChan2 != nil {		num1 := <- intChan2		fmt.Println("num1:",num1)	}	// 報錯	// intChan2 <- 30}

管道的阻塞

【1】當管道只寫入數(shù)據(jù),沒有讀取,就會出現(xiàn)阻塞:

package mainimport (	"fmt"	"sync")var wg sync.WaitGroupfunc writeData(intChan chan int)  {	defer wg.Done()	for i := 1; i < 10; i++ {		intChan <- i		fmt.Println("寫入的數(shù)據(jù):",i)	}	close(intChan)}func readData(intChan chan int)  {	defer wg.Done()	for v := range intChan {		fmt.Println("讀取的數(shù)據(jù)為:",v)	}}func main()  {	intChan := make(chan int, 10)	wg.Add(2)	go writeData(intChan)	// go readData(intChan)	wg.Wait()}

運行結(jié)果Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

【2】寫的快,讀的慢(管道讀寫頻率不一致),不會出現(xiàn)阻塞問題:

package mainimport (	"fmt"	"sync"	"time")var wg sync.WaitGroupfunc writeData(intChan chan int)  {	defer wg.Done()	for i := 1; i < 10; i++ {		intChan <- i		fmt.Println("寫入的數(shù)據(jù):",i)	}	close(intChan)}func readData(intChan chan int)  {	defer wg.Done()	for v := range intChan {		fmt.Println("讀取的數(shù)據(jù)為:",v)		time.Sleep(time.Second)	}}func main()  {	intChan := make(chan int, 10)	wg.Add(2)	go writeData(intChan)	go readData(intChan)	wg.Wait()}

select功能

【1】select功能:解決多個管道的選擇問題,也可以叫做多路復用,可以從多個管道中隨機公平地選擇一個來執(zhí)行

PS:case后面必須進行的是io操作,不能是等值,隨機去選擇一個io操作Ibr28資訊網(wǎng)——每日最新資訊28at.com

PS:default防止select被阻塞住,加入defaultIbr28資訊網(wǎng)——每日最新資訊28at.com

【2】代碼:

package mainimport (	"fmt"	"time")func main()  {	intChan := make(chan int, 1)	go func ()  {		time.Sleep(time.Second * 15)		intChan <- 15	}()	stringChan := make(chan string, 1)	go func ()  {		time.Sleep(time.Second * 12)		stringChan <- "hellocyz"	}()	//本身取數(shù)據(jù)就是阻塞的	// fmt.Println(<-intChan)	select {		case v := <-intChan : fmt.Println("intChan:",v)		case v := <-stringChan : fmt.Println("stringChan:",v)		default: fmt.Println("防止select被阻塞")	}}

defer+recover機制處理錯誤

【1】問題原因:多個協(xié)程工作,其中一個協(xié)程出現(xiàn)panic,導致程序崩潰

【2】解決辦法:利用defer+recover捕獲panic進行處理,即使協(xié)程出現(xiàn)問題,主線程仍然不受影響可以繼續(xù)執(zhí)行。

【3】案例:

package mainimport (	"fmt"	"time")//輸出數(shù)字:func printNum()  {	for i := 1; i <= 10; i++ {		fmt.Println(i)		}}//做除法操作:func divide()  {	defer func ()  {		err := recover()		if err != nil {			fmt.Println("devide()出現(xiàn)錯誤:",err)		}	}()	num1 := 10	num2 := 0	result := num1 / num2	fmt.Println(result)}func main()  {	//啟動兩個協(xié)程:	go printNum()	go divide()	time.Sleep(time.Second * 5)}

結(jié)果:Ibr28資訊網(wǎng)——每日最新資訊28at.com

Ibr28資訊網(wǎng)——每日最新資訊28at.com

本文鏈接:http://www.tebozhan.com/showinfo-26-5170-0.html聊聊協(xié)程和管道—管道

聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 如何使用Kafka構(gòu)建事件驅(qū)動的架構(gòu)?

下一篇: 空間智能化為產(chǎn)業(yè)轉(zhuǎn)型帶來新動能,華為開放能力與合作伙伴共贏時代

標簽:
  • 熱門焦點
Top