channel知识点总结

channel

创建方式:

ch := make(chan int, 10) 后面的数字代表缓冲队列的容量,如果容量为0则代表非缓冲通道

channel类型的值本身就是并发安全的,一个channel就是一个先进先出的队列。

常用操作
for - range遍历

需要不断从channel中读取数据,当channel关闭时,for循环自动退出,这样一来可以防止读取已关闭channel(读取已关闭channel取到的值为存储数据类型的零值)

func main() {
  sign := make(chan int)
  go func() {
    for x := range sign {
      fmt.Println(x)
    }
  }()
  sign <- 1

  time.Sleep(5*time.Second)
}
struct{}在channel中的运用

struct{}作为占位值来使用,我们使用channel的目的如果是为了实现同步的效果,可以定义一个struct

{}类型的channel,占用的内存几乎可以不计

select处理多个channel

使用select可以同时监控多个通道的情况,只处理未阻塞的分支(可以使用的场景:收到某种信号直接退出)

select{
case sign <- 1:
    return
case <- finish:
    return
}
close

通知关闭所有下游协程

close(sign)

所有读sign的协程都会收到close(sign)的信号,进而退出

值为nil的channel

如果我们仅定义通道 ,而不用make进行初始化,该通道的值就为nil,对于一个值为nil的通道,对它的任何接收和发送操作,都会被一直阻塞。

缓冲通道及非缓冲通道
缓冲通道

元素发送到channel,会按照发送顺序依次排入缓冲队列,如果缓冲队列已满,则对该通道的所有发送操作都会被阻塞,并且发送操作所在的goroutine会顺序地进入通道内部的等待队列,直到通道中有元素被取走,等待队列头部的goroutine才会继续执行发送操作。

同理,当缓冲队列为空的时候,如果还要从channel里接收元素,那么该接收操作也会被阻塞,所在的goroutine同样会放入等待队列。

非缓冲通道

在非缓冲通道中,元素的发送方和接收方对接上,该元素才会被传递。无论发送操作还是接收操作,开始执行的时候就被阻塞,直到配对的操作开始执行。

错误用法:

func main() {
  sign := make(chan int)
  sign <- 1
  <- sign
}

上面的代码在执行到 sign <- 1的时候就被阻塞,需要等待配对的读操作,然而读操作在同一个goroutine里,不会执行到下面的代码,于是造成了死锁。

要传递的元素在大部分情况下,会先从发送方复制 到缓冲通道,之后再由缓冲通道复制给接收方,并删除在通道中该元素的副本。不过,当发送操作在执行时,通道为空,而且刚好存在配对的接收操作,发送方会直接将该元素的副本传递给接收方。

引发panic的情况

对于一个已关闭的channel,如果我们继续给它发送元素,或者我们尝试关闭一个已经关闭的channel,会引发panic。

接收表达式:x, ok <- ch1 。

其中ok为bool类型,当ok为false的时候,代表channel已关闭,并且无元素可取。但是当ok为true的时候,channel也可能是关闭的,这种情况下,在channel中有还未被取出的元素。

所以,我们需要在发送方来进行关闭channel的操作,在接收方来关闭会存在引发panic的风险。

两个goroutine交替打印数组
package main

import (
  "fmt"
)

func main() {
  arr := []int{1, 2, 3, 4, 5, 6}
  printer(arr)
  //time.Sleep(time.Second*100)
}

func printer(arr []int){
  finish := make(chan struct{}, 2)
  allow := make(chan struct{})
  go func() {
    defer func() {
      finish <- struct{}{}
    }()
    for i := 0; i < len(arr); i++ {
     allow <- struct{}{}
      if i % 2 == 0 {
        fmt.Printf("THE FIRST GOROUTINE PRINT index = %d value = %d \n", i, arr[i])
     
      }
    }
  }()

  go func() {
    defer func() {
      finish <- struct{}{}
    }()
    for i := 0; i < len(arr); i++ {
     <- allow
      if i % 2 == 1 {
        fmt.Printf("THE SECOND GOROUTINE PRINT index = %d value = %d \n", i, arr[i])
      }
    }
  }()

  <- finish
  <- finish
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值