Go 语言怎么通过通信共享内存?

简介: Go 语言怎么通过通信共享内存?

介绍

Go 语言使用 goroutinechannel,可以实现通过通信共享内存。

本文我们介绍 Go 语言怎么通过通信共享内存。

goroutinechannel

在了解 Go 语言怎么通过通信共享内存之前。我们需要先了解一些预备知识,即 goroutinechannel 是什么?

goroutine 具有简单的模型:它是与其它 goroutine并发运行在同一地址空间的函数。

goroutine 是轻量级的,所有消耗几乎就只有栈空间的分配。而且栈最开始是非常小的,所以他们很廉价,仅在需要时才会随着堆空间的分配(和释放)而变化。

摘自「Effective Go - channels[1]」。

注意:goroutine 之所以取名为 goroutine,是因为现有的术语 - 线程、协程、进程等等 - 无法准确传达它的含义。也有些资料将 goroutine 翻译为 Go 协程或 Go 程。

使用 goroutine 也非常简单,在函数或方法前添加 go 关键字,即可在新的 goroutine 中调用它。当调用完成后,该 goroutine 也会安静地退出。

此外,匿名函数也可以在 goroutine 中调用。

关于 goroutine 的实现原理和调度器模型 GPM,感兴趣的读者朋友们可以自行查阅相关资料。

我们已了解,什么是 goroutine,以及怎么使用 goroutine 调用函数或方法、匿名函数。

但是,想要实现 goroutine 之间的通信,我们还需要了解 channel

channel 需要使用内置函数 make 分配内存,其结果值充当了对底层数据结构的引用。如果提供了一个可选的参数,它就会为该 channel 设置缓冲区大小,否则,该 channel 则为无缓冲区的 channel

关于 channel 的实现原理,感兴趣的读者朋友们可以阅读「Golang 语言中的 channel 实现原理」。

需要注意的是,两个 goroutine 之间通过无缓冲区的 channel 通信时,同步交换数据。

作为两个 goroutine 之间的通信管道,向 channel 中发送数据的 goroutine 称为“发送者”,反之,从 channel 中接收数据的 goroutine 称为“接收者”。

03

通过通信共享内存

我们已经基本了解 Go 语言的 goroutinechannel,接下来我们看一下两个 goroutine 之间怎么使用 channel (无缓冲区和缓冲区)进行通信?

示例代码:

func main() {
 c := make(chan int) // 定义一个无缓冲区 channel
 go func() {         // 启动一个 goroutine 调用匿名函数
  fmt.Println("启动一个 goroutine 调用匿名函数")
  c <- 1 // 该 goroutine 向 channel 发送一个值(信号)
 }()
 fmt.Println("main 函数")
 out := <-c // main goroutine 从 channel 中接收一个值(信号),再未接收到值(信号)之前,一直阻塞
 fmt.Println(out) // 该打印无实际意义,仅为了读者容易理解
}

阅读上面这段代码,我们定义一个无缓冲区 channel,执行匿名函数的 goroutine 作为发送者,main goroutine 作为接收者。

需要注意的是,无缓冲区 channel,接收者在收到值之前,发送者会一直阻塞。同理,发送者在发送值之前,接收者也会一直阻塞。

示例代码:

func main() {
  // c := make(chan int) // 无缓冲区 channel
 c := make(chan int, 5) // 缓冲区 channel
 for i := 0; i < 20; i++ {
  c <- 1
  go func() {
   fmt.Println("do something:", i)
   <-c
  }()
 }
 time.Sleep(time.Second * 2) // 为了防止 main goroutine 提前退出
}

阅读上面这段代码,我们定义一个缓冲区大小为 5 的 channel,执行匿名函数的 goroutine 作为接收者,main goroutine 作为发送者。

需要注意的是,该段代码中有 5 个执行匿名函数的 goroutine,即 N 个接收者,1 个发送者(main goroutine)。

我们前面讲过,接收者在收到值之前会一直阻塞,而无缓冲区 channel 在接收者收到值之前,发送者会一直阻塞。

如果我们将上面这段代码中的缓冲区 channel 换成无缓冲区 channelN - 1 个接收者在接收到值之前,发送者会一直阻塞,发送者阻塞,导致接收者一直接收不到值,也会一直阻塞,从而导致死锁。

上面这段话有些拗口,读者朋友们可以通过运行使用无缓冲区 channel 的代码来帮助自己理解。

我们运行使用缓冲区大小为 5 的 channel 的代码,发现代码可以正常运行,发送者和接收者之间不会产生死锁。

这是因为缓冲区 channel,发送者仅在值被复制到缓冲区之前阻塞,如果缓冲区已满,发送者会一直阻塞,直到某个接收者取出一个值。

回到上面这段示例代码中,执行匿名函数的 N 个 goroutine 作为接收者,在没有收到 main goroutine 发送的数据之前,一直处于阻塞状态,直到作为发送者的 main goroutine 发送数据到缓冲区 channel 中。

读者朋友们如果仔细阅读这段代码,会发现上面这段代码虽然不会产生死锁,但是存在一个 bug

解决方案可以阅读我们之前的一篇文章「Go 语言使用 goroutine 运行闭包的“坑”」,限于篇幅,我就不在本文中赘述了。

04

总结

本文我们介绍 Go 语言中,什么是 goroutinechannel,其中 channel 分为无缓冲区和缓冲区。

在简单了解 goroutinechannel 后,我们又介绍怎么使用 channel,实现两个 goroutine 之间通信。


目录
相关文章
|
9天前
|
监控 算法 Go
Golang深入浅出之-Go语言中的服务熔断、降级与限流策略
【5月更文挑战第4天】本文探讨了分布式系统中保障稳定性的重要策略:服务熔断、降级和限流。服务熔断通过快速失败和暂停故障服务调用来保护系统;服务降级在压力大时提供有限功能以保持整体可用性;限流控制访问频率,防止过载。文中列举了常见问题、解决方案,并提供了Go语言实现示例。合理应用这些策略能增强系统韧性和可用性。
40 0
|
1天前
|
存储 编译器 Go
Go语言学习12-数据的使用
【5月更文挑战第5天】本篇 Huazie 向大家介绍 Go 语言数据的使用,包含赋值语句、常量与变量、可比性与有序性
37 6
Go语言学习12-数据的使用
|
3天前
|
Java Go
一文带你速通go语言指针
Go语言指针入门指南:简述指针用于提升效率,通过地址操作变量。文章作者sharkChili是Java/CSDN专家,维护Java Guide项目。文中介绍指针声明、取值,展示如何通过指针修改变量值及在函数中的应用。通过实例解析如何使用指针优化函数,以实现对原变量的直接修改。作者还邀请读者加入交流群深入探讨,并鼓励关注其公众号“写代码的SharkChili”。
9 0
|
3天前
|
存储 缓存 Java
来聊聊go语言的hashMap
本文介绍了Go语言中的`map`与Java的不同设计思想。作者`sharkChili`是一名Java和Go开发者,同时也是CSDN博客专家及JavaGuide项目的维护者。文章探讨了Go语言`map`的数据结构,包括`count`、`buckets指针`和`bmap`,解释了键值对的存储方式,如何利用内存对齐优化空间使用,并展示了`map`的初始化、插入键值对以及查找数据的源码过程。此外,作者还分享了如何通过汇编查看`map`操作,并鼓励读者深入研究Go的哈希冲突解决和源码。最后,作者提供了一个交流群,供读者讨论相关话题。
14 0
|
4天前
|
Java Go
Go语言学习11-数据初始化
【5月更文挑战第3天】本篇带大家通过内建函数 new 和 make 了解Go语言的数据初始化过程
17 1
Go语言学习11-数据初始化
|
4天前
|
自然语言处理 安全 Java
速通Go语言编译过程
Go语言编译过程详解:从词法分析(生成token)到句法分析(构建语法树),再到语义分析(类型检查、推断、匹配及函数内联)、生成中间码(SSA)和汇编码。最后,通过链接生成可执行文件。作者sharkchili,CSDN Java博客专家,分享技术细节,邀请读者加入交流群。
22 2
|
5天前
|
Java Linux Go
一文带你速通Go语言基础语法
本文是关于Go语言的入门介绍,作者因其简洁高效的特性对Go语言情有独钟。文章首先概述了Go语言的优势,包括快速上手、并发编程简单、设计简洁且功能强大,以及丰富的标准库。接着,文章通过示例展示了如何编写和运行Go代码,包括声明包、导入包和输出语句。此外,还介绍了Go的语法基础,如变量类型(数字、字符串、布尔和复数)、变量赋值、类型转换和默认值。文章还涉及条件分支(if和switch)和循环结构(for)。最后,简要提到了Go函数的定义和多返回值特性,以及一些常见的Go命令。作者计划在后续文章中进一步探讨Go语言的其他方面。
10 0
|
6天前
|
JavaScript 前端开发 Go
Go语言的入门学习
【4月更文挑战第7天】Go语言,通常称为Golang,是由Google设计并开发的一种编程语言,它于2009年公开发布。Go的设计团队主要包括Robert Griesemer、Rob Pike和Ken Thompson,这三位都是计算机科学和软件工程领域的杰出人物。
14 1
|
6天前
|
Go
|
7天前
|
分布式计算 Java Go
Golang深入浅出之-Go语言中的分布式计算框架Apache Beam
【5月更文挑战第6天】Apache Beam是一个统一的编程模型,适用于批处理和流处理,主要支持Java和Python,但也提供实验性的Go SDK。Go SDK的基本概念包括`PTransform`、`PCollection`和`Pipeline`。在使用中,需注意类型转换、窗口和触发器配置、资源管理和错误处理。尽管Go SDK文档有限,生态系统尚不成熟,且性能可能不高,但它仍为分布式计算提供了可移植的解决方案。通过理解和掌握Beam模型,开发者能编写高效的数据处理程序。
135 1