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

Golang|Channel 相关用法理解

文章目录

    • 用 channel 作为并发小容器
    • channel 的遍历
    • channel 导致的死锁问题
    • 用 channel 传递信号
    • 用 channel 并行处理文件
    • 用channel 限制接口的并发请求量
    • 用 channel 限制协程的总数量

用 channel 作为并发小容器

在这里插入图片描述
在这里插入图片描述

  • 注意这里的 ok 如果为 false,表示此时不仅channel为空,而且channel已经被关闭了

channel 的遍历

在这里插入图片描述

  • 注意,遍历会使头指针往后移,相当于取走元素
  • 如果 close channel 注释掉,也不会报错,但是会阻塞,导致输出bye bye这一句代码得不到执行
  • 遍历的另外一种写法

在这里插入图片描述

channel 导致的死锁问题

  • 上面的main方法是等3秒钟结束main协程,更好的方法是使用waitGroup

在这里插入图片描述
在这里插入图片描述

  • 如果 close channel 注释掉,程序执行的时候会报错 fatal error: all goroutines are asleep - deadlock!
  • travese和main都阻塞了

用 channel 传递信号

  • 可以用向channel中传递信号,代替waitGroup,优雅地等子协程结束

在这里插入图片描述

  • struct{} 空结构体类型,空结构体实例 struct{}{}
  • 空结构体在go语言里是一种特殊的结构体,go语言通过一个统一的引用变量来表示所有的空结构体,而且不占用任何的内存空间
  • 使用空结构体,语义会更加明确且不占内存

在这里插入图片描述

  • 在 Go 语言(Golang)中,reflect 是一个非常强大的包,提供了 运行时反射机制,可以在运行时检查变量的类型、获取或设置变量的值。
// 获取类型和值

var x int = 42
t := reflect.TypeOf(x) // reflect.Type
v := reflect.ValueOf(x) // reflect.Value

fmt.Println("类型:", t)      // int
fmt.Println("值:", v.Int()) // 42
// 修改变量的值(需要传指针)

var x int = 10
v := reflect.ValueOf(&x)           // 注意要传指针
v.Elem().SetInt(100)               // 修改值
fmt.Println("x的新值:", x)         // 100
// 检查变量类型

func checkType(i interface{}) {
    t := reflect.TypeOf(i)
    switch t.Kind() {
    case reflect.Int:
        fmt.Println("是整数")
    case reflect.String:
        fmt.Println("是字符串")
    default:
        fmt.Println("其他类型")
    }
}
// 结构体字段操作

type Person struct {
    Name string
    Age  int
}

p := Person{"Tom", 30}
v := reflect.ValueOf(p)
t := reflect.TypeOf(p)

for i := 0; i < t.NumField(); i++ {
    field := t.Field(i)
    value := v.Field(i)
    fmt.Printf("%s: %v\n", field.Name, value)
}

用 channel 并行处理文件

  • channel 不仅可以当作数据容器使用,也可以当作信号容器来使用
  • 下面这个例子是要把多个txt文件合并为一个txt文件,一个常规的思路就是我们顺序读取文件,每读取一行就把这一行对应地写入到新的文件里面去,但是io操作很消耗时间,且三个文件毫不相关,怎么加速?
  • 考虑三个goroutine并行读,且往一个buffer channel写数据,由一个goroutinue往buffer channel里读数据写入文件,这样可以协调读者和写者的速度不匹配问题

在这里插入图片描述

在这里插入图片描述

  • 初始化pc_sync里面有3个元素,协程完成后会从pc_sync取走一个元素,当所有协程结束后,pc_sync为空

在这里插入图片描述
在这里插入图片描述

用channel 限制接口的并发请求量

  • 有时候我们需要限制微服务接口的并发请求度,因为有些接口会涉及到大量的cpu计算或者是内存开销,如果瞬间并发度太大的话,服务器负载会很高
    在这里插入图片描述
  • 但是如果限制了的话必然会牺牲用户的体验
  • 用channel的阻塞机制实现

在这里插入图片描述

用 channel 限制协程的总数量

  • 需要封装,对于所有的协程创建都去走一个统一的入口

在这里插入图片描述

  • ticker 实际上就是每隔一段时间会给ticker.C中放入一个元素

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

相关文章:

  • 大模型SAM辅助labelme分割数据集(纯小白教程)
  • Java栈与队列深度解析:结构、实现与应用指南
  • 用密钥方式让通过JumpServer代理的服务器可以在我本地电脑直接访问
  • Java 设计模式:外观模式详解
  • 5.6 GitHub PR分析爆款方案:分层提示工程+LangChain实战,准确率飙升22%
  • 什么是RAG
  • Nodejs Express框架
  • 【ai回答记录】在sql中使用DATE_SUB 跟 用python或者java的Date计算时间差,哪个速度更加快?
  • 214、【数组】下一个排列(Python)
  • 目标追踪数据标注
  • WITRAN_2DPSGMU_Encoder 类
  • Lombok库
  • 电子电器架构 --- 智能座舱的定义
  • [Linux]umask及其设置
  • 【prometheus+Grafana篇】Prometheus与Grafana:深入了解监控架构与数据可视化分析平台
  • 深度学习总结(12)
  • 10.第二阶段x64游戏实战-添加计时器
  • Mysql概述
  • MCP 认证考试常见技术难题实战分析与解决方案
  • Python(14)Python内置函数完全指南:从基础使用到高阶技巧
  • 常熟银行一季度净赚超10亿增逾13%,净息差较上年末下降0.1个百分点
  • 韩国京畿道骊州市市长率团访问菏泽:想和菏泽一起办牡丹节
  • 由重商主义观察世界现代化历程
  • 因商标近似李小龙形象被裁定无效,真功夫起诉国家知产局,法院判了
  • 涉军民事案件类型日益增多,最高法新规明晰管辖争议问题
  • 济南市莱芜区委书记焦卫星任济南市副市长