首页 > 其他分享 >Go 快速入门指南 - 缓冲通道和非缓冲通道

Go 快速入门指南 - 缓冲通道和非缓冲通道

时间:2022-12-24 18:33:37浏览次数:54  
标签:ch 缓冲 make goroutine 阻塞 Go 接收 通道

概述

建议先阅读 goroutine 小节。

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

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

阻塞通道与非阻塞通道

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

阻塞通道


Go 快速入门指南 - 缓冲通道和非缓冲通道_Go

图片来源: 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
*/

非阻塞通道

Go 快速入门指南 - 缓冲通道和非缓冲通道_Go_02

图片来源:​​https://stackoverflow.com/questions/39826692/what-are-channels-used-for​

例子

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

3 种操作

发送

  • • 如果通道已满 (元素数量达到容量), 发送操作将会阻塞,直到另一个 ​​goroutine​​ 在对应的通道上面完成接收操作, 两个 ​​goroutine​​ 才可以继续执行
  • • 如果通道未满,发送操作不会阻塞

语法规则

通道变量 <- 数据

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

接收

  • • 如果通道已空 (元素数量为 0),接收操作将会阻塞,直到另一个 ​​goroutine​​ 在对应的通道上面完成发送操作, 两个 ​​goroutine​​ 才可以继续执行
  • • 如果通道不为空,接收操作不会阻塞

语法规则

<- 通道变量

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

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

关闭

详情见 关闭通道。

例子

缓存通道容量为 2

package main

func main() {
ch := make(chan string, 2)

ch <- "hello" // 不会死锁,因为 ch 是缓冲通道
ch <- "world"

println(<-ch)
println(<-ch)
}

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

扩展阅读

  1. 1. 死锁 - 维基百科

Go 快速入门指南 - 缓冲通道和非缓冲通道_Go_03

标签:ch,缓冲,make,goroutine,阻塞,Go,接收,通道
From: https://blog.51cto.com/u_15915979/5967442

相关文章

  • Go 快速入门指南 - 通道方向和关闭通道
    概述建议先阅读 阻塞通道 和 非阻塞通道 小节。在前面的两个小节中,为了最小化代码达到演示效果,省略了 ​​关闭通道​​ 的步骤,正确的做法应该是在通道使用完成后关......
  • Go 快速入门指南 - 遍历通道
    概述建议先阅读 range, 阻塞通道, 非阻塞通道 等小节。​​range​​ 除了可以遍历字符串、切片、数组等数据结构外,还可以遍历通道。语法规则和遍历其他数据结构不同,遍......
  • Go 快速入门指南 - 互斥锁和定时器
    互斥锁对于任一共享资源,同一时间保证只有一个操作者,这种方法称为 ​​互斥机制​​。关键字 ​​Mutex​​​ 表示互斥锁类型,它的 ​​Lock​​​ 方法用于获取锁,​​U......
  • django中间件
    django中间件相当于django框架的保安,只要数据过来必须经过中间件的校验。django中间件默认有7个。可以在setings内MIDDLEWARE里查看看着是字符串其实底层就是导模块的......
  • Gogs.02.构建本地仓库并上传到远程仓库
    01.通过VSCode构建一个Vue项目,npminitvue@latest配置参数并安装相关依赖  安装完成后,试运行一下,可以看到已还跑起来了; 02.本地仓构......
  • 使用PicGo+阿里云OSS实现md文档图片上传
    使用PicGo+阿里云OSS实现md文档图片上传这次给大家带来的是PicG0+阿里云Oss+typora的图床环境搭建,帮助大家提高工作效率+写博客速度!1.typora安装给大家一个链接:typora,打......
  • django中间件了解的三个方法、基于Django的插拔式设计、cookie与session简介、基于Dja
    目录Django中间件三个了解的方法1.process_view2.process_exception3.process_template_response基于Django中间实现功能的插拔式设计模拟编写一个消息通知功能(微信、qq、......
  • Go
    Chapter01编译器是全文翻译,解释器是实时翻译。1.环境搭建1.1MAC&Linux下载编译器:MAC:在golang.google.cn中点击DownloadGo,选择ApplemacOS即可,默认安装路径......
  • 1005.Django自定义过滤器及标签
    一、关于自定义自定义的引入内置函数--------->自定义函数内置模块--------->自定义模板内置过滤器------>自定义过滤器内置标签--------->自定义标签二、文件路径配......
  • 在Dubbo-go中使用TLS加密进行安全通信
    1背景Dubbo-go在Getty/Triple/Grpc三个通信层面支持TLS链路安全通信。2原理2.1证书机制:ps:可以先提前了解非对称加密机制。CA(CertificationAuthority)负责生成根证书......