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

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

时间:2022-12-22 16:59:23浏览次数:64  
标签:ch 缓冲 make goroutine chan 阻塞 Go 通道

概述

建议先阅读 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
*/

非阻塞通道

图片

图片来源: 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. 死锁 - 维基百科

联系我

公众号

标签:ch,缓冲,make,goroutine,chan,阻塞,Go,通道
From: https://www.cnblogs.com/duanbiaowu/p/16998868.html

相关文章

  • Go 快速入门指南 - goroutine
    概述goroutine是Go程序并发执行的实体,对于初学者来讲,可以简单地将 goroutine 理解为一个 超轻量的线程。当一个程序启动时,只有一个goroutine调用main函数,称为......
  • Go 快速入门指南 - 类型比较
    概述比较运算符 用来比较两个操作数并返回一个 bool 值,常见的比较运算符:==    等于!=    不等于<     小于<=    小于等于>     大于>=......
  • Go 快速入门指南 - 通道方向和关闭通道
    概述建议先阅读 阻塞通道 和 非阻塞通道 小节。在前面的两个小节中,为了最小化代码达到演示效果,省略了 关闭通道 的步骤,正确的做法应该是在通道使用完成后关闭。......
  • Go 快速入门指南 - 互斥锁和定时器
    互斥锁对于任一共享资源,同一时间保证只有一个操作者,这种方法称为 互斥机制。关键字 Mutex 表示互斥锁类型,它的 Lock 方法用于获取锁,Unlock 方法用于释放锁。在 Lo......
  • Go 快速入门指南 - 遍历通道
    概述建议先阅读 range, 阻塞通道, 非阻塞通道 等小节。range 除了可以遍历字符串、切片、数组等数据结构外,还可以遍历通道。语法规则和遍历其他数据结构不同,遍历......
  • Go 快速入门指南 - 结构体
    概述结构体 是将零个或多个字段(变量)组合在一起的复合数据类型,类似于面向对象语言中的 对象。结构体以及其字段都使用 可见性 规则。语法规则type 结构体名称......
  • Go 快速入门指南 - 嵌套结构体
    概述Go支持将多个结构体通过嵌套的方式,组成一个大的结构体,降低了单个结构体复杂度,同时提高了结构体之间组合的灵活性。例子为了省略篇幅,本小节只使用 字面量 方式初......
  • Go 快速入门指南 - 函数
    概述函数 是将一个或者一类问题包装为一个代码块,可以被多次调用,提高代码重用性。Go函数中声明、定义、参数、返回值这些基础概念,和其他编程语言中的一致,这里不再赘述。......
  • Go 快速入门指南 - 接口和方法
    接口Go接口是隐式实现。 对于一个数据类型,无需声明它实现了哪些接口,只需要实现接口必需的方法即可。当然了,存在一个小问题就是:我们可能无意间实现了某个接口:),所以 ......
  • Go 快速入门指南 - init 函数
    概述init()函数 是一个特殊的函数,一般称为初始化函数,不能被调用。 在每个文件里面,当程序启动或者文件被作为包引用的时候,init()函数就会自动执行,一般用来做一些包的......