Channel

概述 #

建议先阅读 goroutine 小节。

Go 箴言: 不要通过共享内存来通信,而要通过通信来共享内存。

goroutine 是 Go 程序并发执行的实体,channel (通道) 则是它们之间的连接,用于多个 goroutine 之间互相通信。 通道可以让一个 goroutine 发送特定类型值到另一个 goroutine,每一个通道可以发送数据类型称为通道的 元素类型

阻塞通道与非阻塞通道 #

通过关键字 chan + 数据类型 来表明通道数据类型,调用 make() 函数来初始化一个通道。 make() 函数的第二个参数为通道长度,如果未指定或指定为 0,则该通道为非缓存通道 (阻塞通道), 否则该通道为缓存通道 ( 非阻塞通道)。

阻塞通道 #

https://stackoverflow.com/questions/39826692/what-are-channels-used-for

例子 #

ch := make(chan string) // 非缓冲通道
ch := make(chan string, 0) // 非缓冲通道
ch := make(chan string, 10) // 缓冲通道, 容量为 10

3 种操作 #

发送 #

无缓冲通道上面的发送操作将会阻塞,直到另一个 goroutine 在对应的通道上面完成接收操作,两个 goroutine 才可以继续执行。

语法规则 #

通道变量 <- 数据

# 例如: 将变量 x 发送到通道 ch
ch <- x 

接收 #

无缓冲通道上面的接收操作将会阻塞,直到另一个 goroutine 在对应的通道上面完成发送操作,两个 goroutine 才可以继续执行。

语法规则 #

<- 通道变量

# 例如: 从通道 ch 接收一个值,并且丢弃
<-ch 
接收变量 <- 通道变量

# 例如: 从通道 ch 接收一个值,并且赋值给变量 x
x := <-ch 

关闭 #

详情见 关闭通道

例子 #

搭配 goroutine #

package main

func main() {
	ch := make(chan string) // 没有设置通道的长度

	go func() {
		ch <- "hello world"
	}()

	msg := <-ch // 一直阻塞,直到接收到通道消息
	println(msg)
}

// $ go run main.go
// 输出如下
/**
  hello world
*/

死锁 #

package main

func main() {
	ch := make(chan string) // 没有设置通道的长度

	ch <- "hello world" // 向通道发送数据,但是没有接收者

	msg := <-ch // 代码执行不到这里, 因为上面阻塞发送数据时,就已经死锁了
	println(msg)
}

// $ go run main.go
// 输出如下
/**
  fatal error: all goroutines are asleep - deadlock!

  ...
  ...

  exit status 2
*/

扩展阅读 #

  1. 死锁 - 维基百科

转载申请

本作品采用 知识共享署名 4.0 国际许可协议 进行许可,转载时请注明原文链接,图片在使用时请保留全部内容,商业转载请联系作者获得授权。

© 蛮荆 | 陕公网安备 61011302001681 号 | 陕ICP备2023004378号-1 | Rendered by Hugo