当前位置: 首页 > article >正文

golang select两个channel性能稳定,三个channel时性能会发生抖动,为什么?

golang select两个channel性能稳定,三个channel时性能会发生抖动,为什么?

答题思路
select —> 让 Goroutine同时等待多个 Channel 可读或者可写 —> Goroutine —> 调度器调度 —> 资源竞争 —> 不稳定、抖动

在 Go 中,select 语句用于在多个通道操作中进行选择。当有多个通道准备好发送或接收数据时,select 会随机选择一个可用的通道操作执行,当select语句中的通道数量增加时,可能会导致性能方面的变化,尤其是在高并发场景下。

性能稳定与性能抖动的问题通常与调度器和 Goroutine 的调度策略有关。在只有两个通道的情况下,Goroutine 可以在两个通道之间轻松切换,这可以保持较为稳定的性能。

但是,在三个或更多通道的情况下,调度器需要更复杂的调度策略来选择下一个要执行的 Goroutine。这可能导致不同的 Goroutine 之间竞争资源,例如 CPU 时间和内存带宽。在高负载情况下,这种竞争可能导致性能抖动,即执行时间的不稳定性。

此外,如果通道的负载不均衡,例如某个通道的写入速度远快于其他通道,或者某个通道的读取速度远慢于其他通道,也会导致性能抖动。这可能会导致某些 Goroutine 被阻塞,而其他 Goroutine 必须等待更长的时间才能执行。

因此,在编写使用select语句的代码时,需要考虑通道的负载情况以及系统的并发性能。通常情况下,使用有限的通道数量,并确保它们的负载大致相等,可以减少性能抖动的可能性。
(为了提高性能和稳定性,可以考虑使用带有缓冲通道或使用扇入/扇出模式来处理多个通道操作。使用带有缓冲的通道可以减少对通道的频繁操作,而扇入/扇出模式可以将多个通道操作合并为一个操作,减少select 的使用次数。)

//Select 和 Channel
//select是与switch相似的控制结构,与switch不同的是,select中虽然也有多个case,但是这些case中的表达式必须都是 Channel 的收发操作。下面的代码就展示了一个包含 Channel 收发操作的select结构:

package main

import "fmt"

func main() {
  c := make(chan int, 1)
  quit := make(chan int, 1)
  fibonacci(c, quit)
}

func fibonacci(c, quit chan int) {
  x, y := 0, 1
  for {
    select {
    case c <- x:
      x, y = y, x+y 
    case quit <- y:
      x, y = x+y, y
    case <-quit:
      fmt.Println("quit", x, y)
      return
    case <-c:
      fmt.Println("c", x, y)
      return
    default:
      fmt.Println("default")
      return
    }   
  }
}

参考
https://zhuanlan.zhihu.com/p/564277331
https://worktile.com/kb/p/46336


http://www.kler.cn/news/234218.html

相关文章:

  • (c语言版)数组去重和排序 题目描述: 给定一个乱序的数组,删除所有的重复元素,使得每个元素只出现一次,并且按照出现的次数从高到低
  • 设计模式-行为型模式(下)
  • 七、热身仪式(Warm-Up Rituals)
  • 《杨绛传:生活不易,保持优雅》读书摘录
  • Github 2024-02-10 开源项目日报Top10
  • Flink面试准备
  • 代码随想录算法训练营第三十三天丨[重看] 最大子数组、1143. 最长公共子序列/15.4 最长公共子序列
  • Android 识别车牌信息
  • Rust 格式化输出
  • TestNG基础教程
  • fast.ai 机器学习笔记(二)
  • JVM-类加载器 双亲委派机制
  • 项目02《游戏-13-开发》Unity3D
  • 在 Windows上恢复删除照片的 4 种有效方法
  • 初步探索Pyglet库:打造轻量级多媒体与游戏开发利器
  • 多维时序 | Matlab实现RF-Adaboost随机森林结合Adaboost多变量时间序列预测
  • hexo 博客搭建以及踩雷总结
  • 面向对象编程:理解其核心概念与应用
  • linux上部署ftp服务
  • MongoDB聚合:$replaceWith
  • visual studio和cmake如何编译dlib库
  • 深入了解 MySQL 数值型函数
  • Linux文本三剑客(2)
  • 【前后端的那些事】2万字详解WebRTC + 入门demo代码解析
  • 发送get请求并且发送请求头(header),java实现
  • linux系统非关系型数据库redis
  • re:从0开始的CSS学习之路 10. 盒子模型的溢出
  • Python OCR 之旅:PaddleOCR 与 pytesseract 比较及应用
  • 02 动力云客之登陆界面
  • MySQL 的UI