你知道 GO 中的 协程可以无止境的开吗?

2023-10-25 09:50
文章标签 go 知道 协程 无止境

本文主要是介绍你知道 GO 中的 协程可以无止境的开吗?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

GO语言天生高并发的语言,那么是不是使用 go 开辟协程越多越好的,那么在 go 里面,协程是不是可以开无限多个呢?

那么我们就一起来看看尝试写写 demo 吧

尝试开辟尽可能多的 协程

写一个 demo ,循环开 1 << 31 - 1 个协程看看会是什么效果

func main() {goroutineNum := math.MaxInt32for i := 0; i < goroutineNum; i++ {go func(i int) {fmt.Println(" i == ", i, "func == ", runtime.NumGoroutine())}(i)}
}

执行后,我人都傻了,直接是没有输出,2 核 1 G 的服务器直接卡死 , 感兴趣的 xdm 可以尝试一波

这里说一下,出现上述现象的原因是:

我们迅速的疯狂开辟协程,又不控制并发数量,那么在那段很短的时间里面,go 程序会尽可能多的占用操作系统资源,直到被操作系统主动杀掉

一旦主协程被杀掉,那么其他的协程也全部 game over , 因为他们占用的资源是用户态的共享资源,一个协程挂掉,是会影响到其他协程的

尝试控制协程数量

咱们实现的方法是,使用 channel ,设置 channel 的缓冲个数来控制实际并发的协程个数,一起来看看是否有效果

func processGo(i int, ch chan struct{}) {fmt.Println(" i == ", i, "func == ", runtime.NumGoroutine())<-ch
}func main() {goroutineNum := math.MaxInt32ch := make(chan struct{}, 5)for i := 0; i < goroutineNum; i++ {ch <- struct{}{}go processGo(i, ch)}
}

效果见如下截图,由于数据打印太长,如下为部分数据

这里我们可以看到,加入并发控制后,效果还是很明显的,至少我的服务器不会被卡死了

通过打印我们可以看出来,总共 6 个协程,其中有 5 个是子协程,1 个是主协程

我们这里,使用 channel 的方式来控制并发,go 协程的创建速度 依赖于 for 循环的速度,而 for 循环的速度是被 channel 控制住了 ,channel 的速度实际上又被实际处理事情的协程的处理速度控制着,因此,我们可以保证在同一个时间内,并发运行的协程总共是 6 个

但是这就够了么, nonono , 我们可以再来看一个例子

  • 我们设置在循环的个数为 10 ,比刚才的值小了很多,代码逻辑保持一致
func main() {goroutineNum := 10ch := make(chan struct{}, 5)for i := 0; i < goroutineNum; i++ {ch <- struct{}{}go processGo(i, ch)}
}

执行程序看效果

# go run main.goi ==  4 func ==  6i ==  5 func ==  6i ==  6 func ==  6i ==  7 func ==  6i ==  8 func ==  6

我们发现输出并不是我们想要的 , 出现这个的原因是主协程 循环 10 次完毕之后,就会马上退出程序,进而子协程也随之退出,这个问题需要解决

尝试加入 sync 同步机制,让主协程等一下子协程

之前我们有分享到 go 中的一个知识点,可以使用 sync 来一起控制同步 , 就是使用 sync.WaitGroup ,不知道 xdm 是否还记得,不记得没关系,咱们今天再使用一遍,看看效果

  • 加入 sync 机制,循环的时候,需要开辟协程时,则 sync.Add
  • 协程结束的时候,sync.Done
  • 主协程循环完毕之后,等待子协程完成自己的事情,使用 sync.Wait
func processGo(i int, ch chan struct{}) {fmt.Println(" i == ", i, "func == ", runtime.NumGoroutine())<-chwg.Done()
}var wg = sync.WaitGroup{}func main() {goroutineNum := 10ch := make(chan struct{}, 5)for i := 0; i < goroutineNum; i++ {ch <- struct{}{}wg.Add(1)go processGo(i, ch)}wg.Wait()
}

上述代码中,我们可以简单理解 sync 的使用, sync.Add 就是添加需要等待多少个子协程结束, sync.Done 就是当前的子协程结束了,减去 1 个协程, sync.Wait 就是等待 子协程的个数最终变成 0 ,则认为子协程全部关闭

运行程序来查看效果

m# go run main.goi ==  4 func ==  6i ==  5 func ==  6i ==  6 func ==  6i ==  7 func ==  6i ==  8 func ==  6i ==  9 func ==  6i ==  0 func ==  5i ==  1 func ==  4i ==  2 func ==  3i ==  3 func ==  2

尝试做的更加可控一些更加优秀一些

我们可以思考一下,上面的逻辑是不停的有协程在创建,也不停的有协程在被销毁,这样还是很耗资源的,我们是否可以固定设置具体的协程在做事情,并且将发送数据和处理数据进行一个分离呢?

就类似于生产者和消费者一样

咱们来尝试写一个 demo

  • 专门写一个函数用于分发任务
  • 分发任务之前先开辟好对应的协程,等待任务进来
func processGo(i int, ch chan struct{}) {for data := range ch {fmt.Println(" i == ", data, "func == ", runtime.NumGoroutine())wg.Done()}
}func distributeTask(ch chan struct{}) {wg.Add(1)ch <- struct{}{}
}var wg = sync.WaitGroup{}func main() {goroutineNum := 2taskNum := math.MaxInt32ch := make(chan struct{})// 先开辟好协程 等待处理数据for i := 0; i < goroutineNum; i++ {go processGo(i, ch)}// 分发事项for i := 0; i < taskNum; i++ {distributeTask(ch)}wg.Wait()
}

此处使用 sync 控制的同步,可以说是 对应的是任务数量, 主协程是等待所有分发的任务数都被完成了,主协程才关闭程序

执行程序查看效果

 go run main.go

程序正常运行没有毛病,这样做的话,我们可以将分发任务和处理任务进行分离,还大大减少了不必要的协程切换

对于如上案例做一个比喻

channel + sync 的案例 :

最上面的第一种案例,就是相当于动态雇佣 5 个工人,有任务的时候,工人就上去做,做完了自己下岗就得了,反正我这里只容纳 5 个工人,且每个工人做完 1 个任务就得走

分发任务和处理数据的任务分离案例 :

最后的这个案例,就是固定的雇佣 2 个工人干活,项目经理就不停的扔任务进行来,这俩人就疯狂的干

xdm ,go 里面不能滥用协程,需要控制好 go 协程的数量

欢迎点赞,关注,收藏

朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力

好了,本次就到这里

技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。

我是阿兵云原生,欢迎点赞关注收藏,下次见~

这篇关于你知道 GO 中的 协程可以无止境的开吗?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/281750

相关文章

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

go基础知识归纳总结

无缓冲的 channel 和有缓冲的 channel 的区别? 在 Go 语言中,channel 是用来在 goroutines 之间传递数据的主要机制。它们有两种类型:无缓冲的 channel 和有缓冲的 channel。 无缓冲的 channel 行为:无缓冲的 channel 是一种同步的通信方式,发送和接收必须同时发生。如果一个 goroutine 试图通过无缓冲 channel

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而

【Go】go连接clickhouse使用TCP协议

离开你是傻是对是错 是看破是软弱 这结果是爱是恨或者是什么 如果是种解脱 怎么会还有眷恋在我心窝 那么爱你为什么                      🎵 黄品源/莫文蔚《那么爱你为什么》 package mainimport ("context""fmt""log""time""github.com/ClickHouse/clickhouse-go/v2")func main(

使用协程实现高并发的I/O处理

文章目录 1. 协程简介1.1 什么是协程?1.2 协程的特点1.3 Python 中的协程 2. 协程的基本概念2.1 事件循环2.2 协程函数2.3 Future 对象 3. 使用协程实现高并发的 I/O 处理3.1 网络请求3.2 文件读写 4. 实际应用场景4.1 网络爬虫4.2 文件处理 5. 性能分析5.1 上下文切换开销5.2 I/O 等待时间 6. 最佳实践6.1 使用 as

Go Select的实现

select语法总结 select对应的每个case如果有已经准备好的case 则进行chan读写操作;若没有则执行defualt语句;若都没有则阻塞当前goroutine,直到某个chan准备好可读或可写,完成对应的case后退出。 Select的内存布局 了解chanel的实现后对select的语法有个疑问,select如何实现多路复用的,为什么没有在第一个channel操作时阻塞 从而导

Go Channel的实现

channel作为goroutine间通信和同步的重要途径,是Go runtime层实现CSP并发模型重要的成员。在不理解底层实现时,经常在使用中对channe相关语法的表现感到疑惑,尤其是select case的行为。因此在了解channel的应用前先看一眼channel的实现。 Channel内存布局 channel是go的内置类型,它可以被存储到变量中,可以作为函数的参数或返回值,它在r

Go 数组赋值问题

package mainimport "fmt"type Student struct {Name stringAge int}func main() {data := make(map[string]*Student)list := []Student{{Name:"a",Age:1},{Name:"b",Age:2},{Name:"c",Age:3},}// 错误 都指向了最后一个v// a

Go组合

摘要 golang并非完全面向对象的程序语言,为了实现面向对象的继承这一神奇的功能,golang允许struct间使用匿名引入的方式实现对象属性方法的组合 组合使用注意项 使用匿名引入的方式来组合其他struct 默认优先调用外层方法 可以指定匿名struct以调用内层方法 代码 package mainimport ("fmt")type People struct{}type Pe