目录
聊聊Go的goroutine和Channel
首页 后端开发 Golang 聊聊Go的并发编程 (一)

聊聊Go的并发编程 (一)

Jul 07, 2021 pm 04:13 PM
go

聊聊Go的goroutine和Channel

  • 前言
  • 一、goroutine
    • 定义
    • 先看案例知道goroutine怎么用
    • 是什么
  • 二、channel
    • 基础用法
    • 将channel作为参数传递
    • 创建多个channel
    • 将channel作为返回值
    • buffer channel
    • channel关闭

    相关文章推荐:《聊聊Go的并发编程 (二)

    前言

在之前学习go语言时,在看到groutine和channel时就直接跳过了。

当时根本没当回事,这么复杂看它干嘛!(当时的心态)

最近在看go的并发编程,发现全是用的这块内容,那么就只能硬着头皮来了,但是你会发现看着看着其实没那么难。

有时候不想看的东西可以先放着,等自己的注意力集中后在进行查看,你会得到意想不到的收获。

今天这篇文章是一个简单的讲解,咔咔也报了一个go的课程,在哪个课程里边看还能不能获取到更多的理解,随后在进行深度的补充。

一、goroutine

定义

  • 给函数前加上go即可
  • 不需要在定义是区分是否是异步函数
  • 调度器在合适的点进行切换,这个点是有很多的,这里只是参考,不保证切换,不能保证在其它地方不会被切换。IO操作、channel、等待锁、函数调用、runtime.Gosched()等。。。
  • 使用race来检测数据访问冲突

先看案例知道goroutine怎么用

先来看一个案例

案例一

这个案例就是一个简单并发执行的代码,在go里边也就是一个关键字go即可。

那么来看一下这段代码会输出什么

代码输出

从上图可以看到这行代码什么都没有输出,直接就退出了,那这到底是什么情况呢?

直接退出的原因,就是因为我们代码中的main和fmt打印是并发执行的,fmt还没来的急打印数据,外层的循环就已经循环结束了,然后就直接退出了。

在go语言中呢!假设一个main函数退出后,会直接杀死所有的goroutine,所以就造成的现象是,goroutine还没来的急打印数据就被退掉了。

那么你是不是会想,要怎么样才能看到打印的数据呢?其实也很简单,就是让main函数执行完成之后不要着急的退出,给一点等待的时间。看案例

输出结果

这次希望出现的结果就显示出来了。

在本案例中开的goroutine是10个,那么改为1000会怎么样呢?

结果显示还是正常显示,就类似与有1000个人在同时打印东西。

那么设置10跟1000有什么关系吗?

对操作系统熟悉的应该都知道,开10个线程没有问题,开100个线程也没什么大的问题,但是已经差不多了。

一般系统开几十个线程就可以了,那么如果要1000个人同时做一件事情就不能用线程来解决了,需要通过异步方式。

但是在go语言中呢!直接使用go关键字即可,就可以并发执行。

接下来就聊聊为什么go就可以同时1000进行打印。

是什么

先来看看协程和线程的区别。

协程你可以理解为轻量级的线程非抢占式多任务处理,由协程主动交出控制权

线程大家应该都知道是可以被操作系统在任何时候进行切换,所以说线程就是抢占式多任务处理,线程是没有控制权,哪怕是一个语句执行到一半都会被操作系统切掉,然后转到其它线程去操作。

那么反之对于协程来说,什么时候交出控制权,什么时候不交出控制权是由协程内部主动决定的,正是因为这种非抢占式,所以被称之为轻量级。

并且多个协程是可以在一个或多个线程上运行的

二、channel

在第一节中了解到,在go中是可以开非常多的goroutine的,那么goroutine之间的双向通道就是channel

双向通道

基础用法

channel使用方法

从上图案例中可以看到可以直接使用make函数来进行创建channel。

第七行、第八行就是往channel中发送数据。

那么这个案例可以运行吗?来试一下

运行结果

可以看到此时已经报错了,错误的意思就是在往channel发送1的时候会发生死锁。

然后在回到之前的那副图。

goroutine和goroutine之间的交互

在上文我们已经说了,channel是goroutine与goroutine之间的一个交互。

但是此时的案例中缺只有一个goroutine,所以还需要一个另一个goroutine来接收它。

现在你应该了解到如何开启一个goroutine了。

开启另一个goroutine

在上图中我们新开启了另一个goroutine,然后用了一个死循环来接受channel发送的值,并将其打印出来。

但是你会发现我们往channel中发送了俩个数据,此时的打印结果却只有一条数据。但总比我们刚开始的好多了,对吧!

那么为什么会发生这样的情况呢?

可以理理代码的执行流程,先往channel发送了一个1,然后循环获取到第一个值并打印。

再往channel发送数据2,但还没来得及打印就直接退出了,这也就造成了只显示了数据1而没有显示数据2的现象。

这个问题通过咔咔的描述你应该已经知道怎么解决了。

那就是给函数channelDome加一个延迟退出的时间即可。

解决最后一条数据无法显示的情况

将channel作为参数传递

在上文中可以看到go后边跟的是一个闭包函数,在这个闭包中使用的c就是使用的外层的c。

那么将这个c使用参数传递可否呢?答案是肯定可以的。

将channel作为参数传递

当然也可以传递其它的参数

传递其它参数

通过上图可以看到不仅仅传递了channel还传递了id参数,同时还可以将代码直接优化为圈住的部分,也就是直接从channel取值。

创建多个channel

创建多个channel

从上图可以看到每个人都有自己的channel,然后进行分发,分发之后每个人都会收到自己的接收到的值并打印出来。

同样你可以看到我们在26行处还新加了一个for循环给channle里边发送数据。

打印结果

从运行结果中你会发现打印的顺序是混乱的,例如receive i 和receve I这俩个值。

此时你会不会有疑问,我们在往channel中发送数据时是按照顺序发送的啊!那么接收时肯定也是按照顺序接收的。

既然非常确定发送数据是按照顺序的,那么问题就只能出现在Printf这里。

因为Printf是存在IO的,goroutine进行调度,那么此时的Printf是乱序的,但是都会将收到的值一一打印出来。

将channel作为返回值

前几节的案例都是通过创建好的channle然后作为参数传递进去的。

那么本节将会把channel作为一个返回值给返回出去。

将channel作为返回值

源码

package mainimport (
	"fmt"
	"time")func createWorker(id int) chan int {
	c := make(chan int)
	go func() {
		for {
			fmt.Printf("Worker %d receive %c\n", id, <-c)
		}
	}()
	return c}func channelDemo() {
	var channels [10]chan int
	for i := 0; i < 10; i++ {
		channels[i] = createWorker(i)
	}

	for i := 0; i < 10; i++ {
		channels[i] <- 'a' + i	}
	time.Sleep(time.Millisecond)}func main() {
	channelDemo()}
登录后复制

从这里你可以看到我们将worker函数改为了createWorker函数,因为在这个函数里边就是直接创建channel。

接着通过一个协程将channel接收到的值进行打印。

在把channel进行返回出去。

来看一下运行结果

运行结果

通过运行结果可以得知我们的代码编写还是对的,但是此时返回的channel你可以非常直观的看到怎么用

但如果代码数量多的时候,你根本不清楚这个channel怎么用,所有这段代码还需要简单的修饰一下。

那么就需要做的事情的是告诉外面用的人应该怎么用。

给channel发送数据

通过上述代码可以得知,是往channel中发送数据的,那么在createWorker方法的返回的channel要标记一下

修改代码

所以说现在的代码就变成这个样子,我们直接给createWorker方法的返回值channel标记好方向。作用是送数据的。

那么在打印的时候就是收据,这样看起来就非常直观了。

当修改完上面俩步之后你会发现createWorker调用是报错了,Cannot use 'createWorker(i)' (type chan<- int) as type chan int看到错误就应该知道是俩边类型不对等。

修改传入createWorker方法的类型

修改完之后你就会发现编译正确了,没有报错信息了。

运行结果也是ok的。

运行结果

本小节源码

package mainimport (
	"fmt"
	"time")func createWorker(id int) chan<- int {
	c := make(chan int)
	go func() {
		for {
			fmt.Printf("Worker %d receive %c\n", id, <-c)
		}
	}()
	return c}func channelDemo() {
	var channels [10]chan<- int
	for i := 0; i < 10; i++ {
		channels[i] = createWorker(i)
	}

	for i := 0; i < 10; i++ {
		channels[i] <- 'a' + i	}
	time.Sleep(time.Millisecond)}func main() {
	channelDemo()}
登录后复制

buffer channel

学习了这么久了,那么咔咔问你一个问题,这段代码执行会发生什么?

问题代码

没错,会发生报错,因为在文章开头咔咔就讲过了,给一个channel发送数据,就需要开启另一个协程来收数据。

虽然协程我们说是轻量级的,但是如果发送了数据之后,就需要切换协程来进行收数据就非常的耗费资源。

那么就是本节给大家讲解的东西。

buffer channel

创建了可以有3个缓冲区的channel,然后往channel发送3个数据。

同时运行结果也可以得知没有在发生deadlock

给你一个问题,如果往缓冲区在发送一个数据4会发生什么呢?

运行结果

聪明的你,肯定就想到结果了,没错,报了deadlock

接着我们就使用之前的worker,来接受channel的数据。

接收数据

但是你会发现运行结果还是没有打印出送进去的1,2,3,4。

这个问题现在也已经说了好几次了,你可以试着问一下你自己,这种情况你应该怎么解决。

解决代码

也就是加一个延迟时间即可,这里顺便给大家说明一下,之前的那个案例打印的是字母,所有格式化用的%c,现在打印的是数字,所以改为了%d,一点小小的改动。

这种方式建立channel对性能的提升是有一定的作用的。

到现在你有没有发现一个问题,那就是在发送channel时不知道什么时候发完了。

接下来就来看这个问题。

channel关闭

借用上个案例的代码来继续进行说明。

关闭channle
跟上节代码不一致的是,我们在结尾处添加了close,close需要注意的是在发送方进行关闭的。

你会看到运行结果并不如意,你会发现虽然收到了1,2,3,4。

但是下面还接收到了非常多的0,只是截图只截到了一条数据而已。

虽然发送方将channel给close掉了,但是接受放也就是worker还是会收到数据的,不是说channel给close后就收不到数据了。

但是当发送方将channle设置为close之后,收到的数据就都是0,也就是收到的是worker方法传递的c chan int这个参数0的值。

现在我们的channel是一个int类型,收到的是0。那么如果是一个string类型,收到的就是一个空字符串。

这个会收多久呢?也就是咱们设置的一毫秒的时间。

如果让你改这段程序你有没有思路呢?如果没有思路就跟这咔咔的节奏一起摇摆。

代码修改

在函数worker中,使用俩个值来进行接收,n就是传递过来的channel c。ok就是判断这个值是否存在。

可以看到运行结果,就不会在出现接收到0的数据了。

除了这种写法还有一种更简单的方式。

循环处理

坚持学习、坚持写作、坚持分享是咔咔从业以来一直所秉持的信念。希望在偌大互联网中咔咔的文章能带给你一丝丝帮助。我是咔咔,下期见。

以上是聊聊Go的并发编程 (一)的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Go WebSocket 消息如何发送? Go WebSocket 消息如何发送? Jun 03, 2024 pm 04:53 PM

在Go中,可以使用gorilla/websocket包发送WebSocket消息。具体步骤:建立WebSocket连接。发送文本消息:调用WriteMessage(websocket.TextMessage,[]byte("消息"))。发送二进制消息:调用WriteMessage(websocket.BinaryMessage,[]byte{1,2,3})。

深入理解 Golang 函数生命周期与变量作用域 深入理解 Golang 函数生命周期与变量作用域 Apr 19, 2024 am 11:42 AM

在Go中,函数生命周期包括定义、加载、链接、初始化、调用和返回;变量作用域分为函数级和块级,函数内的变量在内部可见,而块内的变量仅在块内可见。

如何在 Go 中使用正则表达式匹配时间戳? 如何在 Go 中使用正则表达式匹配时间戳? Jun 02, 2024 am 09:00 AM

在Go中,可以使用正则表达式匹配时间戳:编译正则表达式字符串,例如用于匹配ISO8601时间戳的表达式:^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d+)?(Z|[+-][0-9]{2}:[0-9]{2})$。使用regexp.MatchString函数检查字符串是否与正则表达式匹配。

Golang 与 Go 语言的区别 Golang 与 Go 语言的区别 May 31, 2024 pm 08:10 PM

Go和Go语言是不同的实体,具有不同的特性。Go(又称Golang)以其并发性、编译速度快、内存管理和跨平台优点而闻名。Go语言的缺点包括生态系统不如其他语言丰富、语法更严格以及缺乏动态类型。

Golang 技术性能优化中如何避免内存泄漏? Golang 技术性能优化中如何避免内存泄漏? Jun 04, 2024 pm 12:27 PM

内存泄漏会导致Go程序内存不断增加,可通过:关闭不再使用的资源,如文件、网络连接和数据库连接。使用弱引用防止内存泄漏,当对象不再被强引用时将其作为垃圾回收目标。利用go协程,协程栈内存会在退出时自动释放,避免内存泄漏。

如何在 IDE 中查看 Golang 函数文档? 如何在 IDE 中查看 Golang 函数文档? Apr 18, 2024 pm 03:06 PM

使用IDE查看Go函数文档:将光标悬停在函数名称上。按下热键(GoLand:Ctrl+Q;VSCode:安装GoExtensionPack后,F1并选择"Go:ShowDocumentation")。

Go 并发函数的单元测试指南 Go 并发函数的单元测试指南 May 03, 2024 am 10:54 AM

对并发函数进行单元测试至关重要,因为这有助于确保其在并发环境中的正确行为。测试并发函数时必须考虑互斥、同步和隔离等基本原理。可以通过模拟、测试竞争条件和验证结果等方法对并发函数进行单元测试。

Golang 函数接收 map 参数时的注意事项 Golang 函数接收 map 参数时的注意事项 Jun 04, 2024 am 10:31 AM

在Go中传递map给函数时,默认会创建副本,对副本的修改不影响原map。如果需要修改原始map,可通过指针传递。空map需小心处理,因为技术上是nil指针,传递空map给期望非空map的函数会发生错误。

See all articles