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

詳解Golang中Channel的用法

 更新時(shí)間:2020年11月09日 17:08:03   作者:極速蝸蝸  
如果說goroutine是Go語言程序的并發(fā)體的話,那么channels則是它們之間的通信機(jī)制。這篇文章主要介紹Golang中Channel的用法,需要的朋友可以參考下

如果說goroutine是Go語言程序的并發(fā)體的話,那么channels則是它們之間的通信機(jī)制。一個(gè)channel是一個(gè)通信機(jī)制,它可以讓一個(gè)goroutine通過它給另一個(gè)goroutine發(fā)送值信息。

1 創(chuàng)建channel

每個(gè)channel都有一個(gè)特殊的類型,也就是channels可發(fā)送數(shù)據(jù)的類型。一個(gè)可以發(fā)送int類型數(shù)據(jù)
的channel一般寫為chan int。使用內(nèi)置的make函數(shù),如果第二個(gè)參數(shù)大于0,則表示創(chuàng)建一個(gè)帶緩存的channel。

ch := make(chan int) // ch has type 'chan int'
ch = make(chan int, 3) // buffered channel with capacity 3

2 channel的發(fā)送和接受

一個(gè)發(fā)送語句將一個(gè)值從一個(gè)goroutine通過channel發(fā)送到另一個(gè)執(zhí)行接收操作的goroutine。發(fā)送和接收兩個(gè)操作都使用<-運(yùn)算符。在發(fā)送語句中,<-運(yùn)算符分割channel和要發(fā)送的值。在接收語句中,<-運(yùn)算符寫在channel對象之前。一個(gè)不使用接收結(jié)果的接收操作也是合法的。

ch <- x
// a send statement
x = <-ch // a receive expression in an assignment statement
<-ch
// a receive statement; result is discarded

3 channel的close

Channel還支持close操作,用于關(guān)閉channel,隨后對基于該channel的任何發(fā)送操作都將導(dǎo)致panic異常。對一個(gè)已經(jīng)被close過的channel進(jìn)行接收操作依然可以接受到之前已經(jīng)成功發(fā)送的數(shù)據(jù),如果channel中已經(jīng)沒有數(shù)據(jù)的話將產(chǎn)生一個(gè)零值的數(shù)據(jù)。使用內(nèi)置的close函數(shù)就可以關(guān)閉一個(gè)channel:

close(ch)

4 不帶緩存的Channels

一個(gè)基于無緩存Channels的發(fā)送操作將導(dǎo)致發(fā)送者goroutine阻塞,直到另一個(gè)goroutine在相同的Channels上執(zhí)行接收操作,當(dāng)發(fā)送的值通過Channels成功傳輸之后,兩個(gè)goroutine可以繼續(xù)執(zhí)行后面的語句。反之,如果接收操作先發(fā)生,那么接收者goroutine也將阻塞,直到有另一個(gè)goroutine在相同的Channels上執(zhí)行發(fā)送操作。
基于無緩存Channels的發(fā)送和接收操作將導(dǎo)致兩個(gè)goroutine做一次同步操作。因?yàn)檫@個(gè)原因,無緩存Channels有時(shí)候也被稱為同步Channels。

5 串聯(lián)的Channels

Channels也可以用于將多個(gè)goroutine連接在一起,一個(gè)Channel的輸出作為下一個(gè)Channel的輸入。這種串聯(lián)的Channels就是所謂的管道(pipeline)。

func main() {
	naturals := make(chan int)
	squares := make(chan int)
	// Counter
	go func() {
		for x := 0; x < 100; x++ {
			naturals <- x
		}
		close(naturals)
	}()
	// Squarer
	go func() {
		for x := range naturals {
			squares <- x * x
		}
		close(squares)
	}()
	// Printer (in main goroutine)
	for x := range squares {
		fmt.Println(x)
	}
}

當(dāng)一個(gè)被關(guān)閉的channel中已經(jīng)發(fā)送的數(shù)據(jù)都被成功接收后,后續(xù)的接收操作將不再阻塞,它們會立即返回一個(gè)零值。
Go語言的range循環(huán)可直接在channels上面迭代。使用range循環(huán)依次從channel接收數(shù)據(jù),當(dāng)channel被關(guān)閉并且沒有值可接收時(shí)跳出循環(huán)。

6 單方向的Channels

為了防止被濫用,Go語言的類型系統(tǒng)提供了單方向的channel類型,分別用于只發(fā)送或只接收的channel。類型<-chan int表示一個(gè)只接收int的channel, chan<- int表示一個(gè)只發(fā)送int的channel,(箭頭<-和關(guān)鍵字chan的相對位置表明了channel的方向。),這種限制將在編譯期檢測。

func counter(out chan<- int) {
	for x := 0; x < 100; x++ {
			out <- x
		}
		close(out)
	}
	func squarer(out chan<- int, in <-chan int) {
		for v := range in {
			out <- v * v
		}
		close(out)
	}
	func printer(in <-chan int) {
		for v := range in {
			fmt.Println(v)
		}
	}
	func main() {
		naturals := make(chan int)
		squares := make(chan int)
		go counter(naturals)
		go squarer(squares, naturals)
		printer(squares)
}

7 帶緩存的Channels

帶緩存的Channel內(nèi)部持有一個(gè)元素隊(duì)列。隊(duì)列的最大容量是在調(diào)用make函數(shù)創(chuàng)建channel時(shí)通過第二個(gè)參數(shù)指定的。
向緩存Channel的發(fā)送操作就是向內(nèi)部緩存隊(duì)列的尾部插入元素,接收操作則是從隊(duì)列的頭部刪除元素。如果內(nèi)部緩存隊(duì)列是滿的,那么發(fā)送操作將阻塞直到因另一個(gè)goroutine執(zhí)行接收操作而釋放了新的隊(duì)列空間。相反,如果channel是空的,接收操作將阻塞直到有另一個(gè)goroutine執(zhí)行發(fā)送操作而向隊(duì)列插入元素。

  • write:緩沖區(qū)被填滿后,寫端才會阻塞。
  • read:緩沖區(qū)被讀空,讀端才會阻塞。

可以用內(nèi)置的cap函數(shù)獲取channel內(nèi)部緩存的容量

fmt.Println(cap(ch)) // "3"

可以用內(nèi)置的len函數(shù)獲取channel內(nèi)部緩存隊(duì)列中有效元素的個(gè)數(shù)。

fmt.Println(len(ch)) // "2"

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

相關(guān)文章

  • golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法

    golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法

    今天小編就為大家分享一篇golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-07-07
  • golang defer執(zhí)行順序全面詳解

    golang defer執(zhí)行順序全面詳解

    這篇文章主要為大家介紹了golang defer執(zhí)行順序全面詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • GoLang channel底層代碼實(shí)現(xiàn)詳解

    GoLang channel底層代碼實(shí)現(xiàn)詳解

    Channel和goroutine的結(jié)合是Go并發(fā)編程的大殺器。而Channel的實(shí)際應(yīng)用也經(jīng)常讓人眼前一亮,通過與select,cancel,timer等結(jié)合,它能實(shí)現(xiàn)各種各樣的功能。接下來,我們就要梳理一下GoLang channel底層代碼實(shí)現(xiàn)
    2022-10-10
  • 淺談goland導(dǎo)入自定義包時(shí)出錯(cuò)(一招解決問題)

    淺談goland導(dǎo)入自定義包時(shí)出錯(cuò)(一招解決問題)

    這篇文章主要介紹了淺談goland導(dǎo)入自定義包時(shí)出錯(cuò)(一招解決問題),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang-切片slice的創(chuàng)建方式

    golang-切片slice的創(chuàng)建方式

    這篇文章主要介紹了golang-切片slice的創(chuàng)建方式,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • golang判斷net.Conn 是否已關(guān)閉的操作

    golang判斷net.Conn 是否已關(guān)閉的操作

    這篇文章主要介紹了golang判斷net.Conn 是否已關(guān)閉的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang中的net/rpc包使用概述(小結(jié))

    golang中的net/rpc包使用概述(小結(jié))

    本篇文章主要介紹了golang中的net/rpc包使用概述(小結(jié)),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-11-11
  • GoLang中Json?Tag用法實(shí)例總結(jié)

    GoLang中Json?Tag用法實(shí)例總結(jié)

    這篇文章主要給大家介紹了關(guān)于GoLang中Json?Tag用法的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2022-02-02
  • Golang學(xué)習(xí)之內(nèi)存逃逸分析

    Golang學(xué)習(xí)之內(nèi)存逃逸分析

    內(nèi)存逃逸分析是go的編譯器在編譯期間,根據(jù)變量的類型和作用域,確定變量是堆上還是棧上。本文將帶大家分析一下Golang中的內(nèi)存逃逸,需要的可以了解一下
    2023-01-01
  • Golang接口型函數(shù)使用小結(jié)

    Golang接口型函數(shù)使用小結(jié)

    接口函數(shù)指的是用函數(shù)實(shí)現(xiàn)接口,這樣在調(diào)用的時(shí)候就會非常簡便,這種方式適用于只有一個(gè)函數(shù)的接口,這里以迭代一個(gè)map為例,演示這一實(shí)現(xiàn)的技巧,對Golang接口型函數(shù)使用知識感興趣的朋友一起看看吧
    2022-06-06

最新評論