[go]channel(介绍3)

juejin.cn/post/743935...

参考网址, 感谢大佬分享

实例

go 复制代码
var ChannelName chan *AAA       //AAA是一个结构体
// 有缓冲的channel, 当通道还有空的时候,就可以一直往里写数据, 如果没空, 就堵塞
ChannelName = make(chan *AAA, 4096) 


func InitChannel() {
    go FUCN_1()

}

func FUCN_1() {
     defer func() {
        err := recover()
        if err != nil {
           glog.Error(err)
        }
     }()

    for {
      request := <-ChannelName
      val, err := json.Marshal(request.val)
      if err == nil {
           db.TraceRedis.Set(request.Key, string(val), time.Hour*24)
      }
    }
}

Go语言中的 Channel 是一种特殊的类型,它提供了一种在两个或多个协程间传递数据的方式。

缓冲区

go 复制代码
ch := make(chan Type, bufferSize)
// 其中,`Type` 是 channel 中传递的元素类型,`bufferSize` 是缓冲区的大小。

在 Go 语言中,channel 是一种用于在不同的 goroutine 之间进行通信的类型化管道。

channel 可以是有缓冲区的(buffered)或无缓冲区的(unbuffered)。

缓冲区的大小决定了 channel 可以同时存储多少个元素而不必阻塞发送方或接收方。

缓冲区大小的选择

选择缓冲区大小通常取决于具体的应用场景和需求:

  1. 无缓冲区(bufferSize == 0

    • 无缓冲区的 channel 意味着发送操作会阻塞,直到另一方准备好接收数据。
    • 适用于需要严格同步的场景,比如信号量、生产者-消费者模型中的同步点。
  2. 有缓冲区

    • 有缓冲区的 channel 允许在缓冲区未满时发送操作不阻塞,或在缓冲区未空时接收操作不阻塞。
    • 缓冲区大小的选择依赖于发送和接收操作的频率和模式。
    • 缓冲区太大会导致内存浪费,太小则可能导致频繁的阻塞和性能问题。

例子

go 复制代码
package main
 
import (
	"fmt"
	"time"
)
 
func producer(ch chan int, count int) {
	for i := 0; i < count; i++ {
		ch <- i
		fmt.Printf("Produced: %d\n", i)
		time.Sleep(time.Millisecond * 100) // 模拟生产时间
	}
	close(ch)
}
 
func consumer(ch chan int) {
	for num := range ch {
		fmt.Printf("Consumed: %d\n", num)
		time.Sleep(time.Millisecond * 150) // 模拟消费时间
	}
}
 
func main() {
	bufferSize := 5
	ch := make(chan int, bufferSize)
 
	go producer(ch, 10)
	go consumer(ch)
 
	time.Sleep(time.Second * 2) // 确保所有操作完成

}

创建

在 Go 语言中,创建一个 channel 非常简单,我们可以使用 make 函数创建一个无缓冲和有缓冲的 channel, 两者的区别在于是否可以存储值。

go 复制代码
ch := make(chan int)     // 创建一个无缓冲 channel
ch := make(chan int, 10) // 创建一个缓冲区位 10 的 channel

读写

(1)阻塞

go 复制代码
ch <- 10  // 写入

val := <-ch // 读取,接受值
<-ch // 读取,不关心值
//当我们往无缓冲的 channel 中写入值时,需要有另外一个协程在读,不然此时写入的协程就会陷于阻塞状态; 
//当我们往一个带有缓冲区的的 channel 中写入值时,如果缓冲区没满时,则可以正常写入,当缓冲区满时,则也会陷入阻塞的状态。

(2)非阻塞

go 复制代码
select {
    case val := <-ch:
            ...
    default:
            ...
}

select {
    case ch <- val:
            ...
    default:
            ...
}
// 尝试从 channel 中读取数据时,使用`select`语句加上`default`分支可实现非阻塞接收。
// 如果 channel 为空,会执行`default`分支的代码。

// 尝试往 channel 中写入数据时,使用`select`语句加上`default`分支可实现非阻塞写入。
// 如果 channel 已满,会执行`default`分支的代码。

3关闭

scss 复制代码
close(ch)

//当你不需要再向 channel 中写入值时,则可以使用 close 函数关闭 channel。
//此时,你还是可以从 channel 中读取值。

//同时,在从 channel 中读取值时,我们还可以根据 channel 的返回值判断 channel 中是否已经为空, 当接受的第二个值为 false 时,则表示 channel 此时已经为空,val 会返回元素类型的零值。
相关推荐
思想者0011 小时前
Chrome访问https页面显示ERR_CERT_INVALID,且无法跳过继续访问
前端·chrome
x_chengqq4 小时前
前端批量下载文件
前端
捕鲸叉6 小时前
QT自定义工具条渐变背景颜色一例
开发语言·前端·c++·qt
傻小胖7 小时前
路由组件与一般组件的区别
前端·vue.js·react.js
Elena_Lucky_baby7 小时前
在Vue3项目中使用svg-sprite-loader
开发语言·前端·javascript
重生之搬砖忍者8 小时前
uniapp使用canvas生成订单小票图片
前端·javascript·canva可画
万水千山走遍TML8 小时前
console.log封装
前端·javascript·typescript·node·log·console·打印封装
阿雄不会写代码8 小时前
使用java springboot 使用 Redis 作为消息队列
前端·bootstrap·html
m0_748236588 小时前
【Nginx 】Nginx 部署前端 vue 项目
前端·vue.js·nginx
@C宝9 小时前
【前端面试题】前端中的两个外边距bug以及什么是BFC
前端·bug