Golang学习笔记:channel

Stella981
• 阅读 444

channel

channel是goroutine之间的通信机制,它可以让一个goroutine通过它给另一个goroutine发送数据,每个channel在创建的时候必须指定一个类型,指定的类型是任意的。
使用内置的make函数,可以创建一个channel类型:

ch := make(chan int)

发送和接受

channel主要的操作有发送和接受:

// 发送数据到channel
ch <- 1
// 从channel接受数据
x := <- ch

如向channel发送数据的时候,该goroutine会一直阻塞直到另一个goroutine接受该channel的数据,反之亦然,goroutine接受channel的数据的时候也会一直阻塞直到另一个goroutine向该channel发送数据,如下面操作:

func main() {
    ch := make(chan string)
    // 在此处阻塞,然后程序会弹出死锁的报错
    c <- "hello"
    fmt.Println("channel has send data")
}

正确的操作:

func main() {
    ch := make(chan string)
    go func(){
        // 在执行到这一步的时候main goroutine才会停止阻塞
        str := <- ch
        fmt.Println("receive data:" + str)
    }()
    ch <- "hello"
    fmt.Println("channel has send data")
}

说到channel的阻塞,就不得不说到有缓冲的channel。

带缓冲的channel

带缓冲的channel的创建和不带缓冲的channel(也就是上面用的channel)的创建差不多,只是在make函数的第二个参数指定缓冲的大小。

// 创建一个容量为10的channel
ch := make(chan int, 10)

带缓冲的channel就像一个队列,遵从先进先从的原则,发送数据向队列尾部添加数据,从头部接受数据。
Golang学习笔记:channel

goroutine向channel发送数据的时候如果缓冲还没满,那么该goroutine就不会阻塞。

ch := make(chan int, 2)
// 前面两次发送数据不会阻塞,因为缓冲还没满
ch <- 1
ch <- 2
// goroutine会在这里阻塞
ch <- 3

反之如果接受该channel数据的时候,如果缓冲有数据,那么该goroutine就不会阻塞。

channel与goroutine之间的应用可以想象成某个工厂的流水线工作,流水线上面有打磨,上色两个步骤(两个goroutine),负责打磨的工人生产完成后会传给负责上色的工人,上色的生产依赖于打磨,两个步骤之间的可能存在存放槽(channel),如果存放槽存满了,打磨工人就不能继续向存放槽当中存放产品,直到上色工人拿走产品,反之上色工人如果把存放槽中的产品都上色完毕,那么他就只能等待新的产品投放到存放槽中。

备注

其实在实际应用中,带缓冲的channel用的并不多,继续拿刚才的流水线来做案例,如果打磨工人生产速度比上色工人工作速度要快,那么即便再多容量的channel,也会迟早被填满然后打磨工人会被阻塞,反之如果上色工人生产速度大于打磨工人速度,那么有缓冲的channel也是一直处于没有数据,上色工人很容易长时间处于阻塞的状态。

因此比较好的解决方法还是针对生产速度较慢的一方多加人手,也就是多开几个goroutine来进行处理,有缓冲的channel最好用处只是拿来防止goroutine的完成时间有一定的波动,需要把结果缓冲起来,以平衡整体channel通信。

单方向的channel

使用channel来使不同的goroutine去进行通信,很多时候都和消费者生产者模式很相似,一个goroutine生产的结果都用channel传送给另一个goroutine,一个goroutine的执行依赖与另一个goroutine的结果。
因此很多情况下,channel都是单方向的,在go里面可以把一个无方向的channel转换为只接受或者只发送的channel,但是却不能反过来把接受或发送的channel转换为无方向的channel,适当地把channel改成单方向,可以达到程序强约束的做法,类似于下面例子:

fuc main(){
    ch := make(ch chan string)
    
    go func(out chan<- string){
        out <- "hello"
    }(ch)
    
    go func(in <-chan string){
        fmt.Println(in)
    }(ch)
    
    time.Sleep(2 * time.Second)
}

select多路复用

在一个goroutine里面,对channel的操作很可能导致我们当前的goroutine阻塞,而我们之后的操作都进行不了。而如果我们又需要在当前channel阻塞进行其他操作,如操作其他channel或直接跳过阻塞,可以通过select来达到多个channel(可同时接受和发送)复用。如下面我们的程序需要同时监听多个频道的信息:

broadcaster1 := make(chan string) // 频道1
broadcaster2 := make(chan string) // 频道2
select {
    case mess1 := <-broadcaster1:
        fmt.Println("来自频道1的消息:" + mess1)
    case mess2 := <-broadcaster2:
        fmt.Println("来自频道2的消息:" + mess2)
    default:
        fmt.Println("暂时没有任何频道的消息,请稍后再来~")
        time.Sleep(2 * time.Second)
}

select和switch语句有点相似,找到匹配的case执行对应的语句块,但是如果有两个或以上匹配的case语句,那么则会随机选择一个执行,如果都不匹配就会执行default语句块(如果含有default的部分的话)。 值得注意的是,select一般配合for循环来达到不断轮询管道的效果,可能很多小伙伴想着写个在某个case里用break来跳出for循环,这是不行的,因为break只会退出当前case,需要使用return来跳出函数或者弄个标志位标记退出

var flag = 0
for {
    if flag == 1 {break}
    select {
        case message := <- user.RecMess :
            event := gjson.Get(string(message), "event").String()
            if event == "login" {
                Login(message, user)
            }
            break
        case <- user.End :
            flag = 1
            break
    }
}
        

关闭

channel可以接受和发送数据,也可以被关闭。

close(ch)

关闭channel后,所有向channel发送数据的操作都会引起panic,而被close之后的channel仍然可以接受之前已经发送成功的channel数据,如果数据全部接受完毕,那么再从channel里面接受数据只会接收到零值得数据。

channel的关闭可以用来操作其他goroutine退出,在运行机制方面,goroutine只有在自身所在函数运行完毕,或者主函数运行完毕才会打断,所以我们可以利用channel的关闭作为程序运行入口的一个标志位,如果channel关闭则停止运行。

无法直接让一个goroutine直接停止另一个goroutine,但可以使用通信的方法让一个goroutine停止另一个goroutine,如下例子就是程序一边运行,一边监听用户的输入,如果用户回车,则退出程序。

func main() {
    shutdown := make(chan struct{})
    var n sync.WaitGroup
    n.Add(1)
    go Running(shutdown, &n) 
    n.Add(1)
    go ListenStop(shutdown, &n) 
    n.Wait()
}

func Running(shutdown <-chan struct{}, n *sync.WaitGroup) {
    defer n.Done()
    for {
        select {
        case <-shutdown:
            // 一旦关闭channel,则可以接收到nil。
            fmt.Println("shutdown goroutine")
            return
        default:
            fmt.Println("I am running")
            time.Sleep(1 * time.Second)
        }   
    }   
}

func ListenStop(shutdown chan<- struct{}, n *sync.WaitGroup) {
    defer n.Done()
    os.Stdin.Read(make([]byte, 1)) 
    // 如果用户输入了回车则退出关闭channel
    close(shutdown)
}

利用channel关闭时候的传送的零值信号可以有效地退出其他goroutine,特别是关闭多个goroutine的时候,就不需要向channel传输多个信息了。

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Karen110 Karen110
2年前
一篇文章带你了解JavaScript日期
日期对象允许您使用日期(年、月、日、小时、分钟、秒和毫秒)。一、JavaScript的日期格式一个JavaScript日期可以写为一个字符串:ThuFeb02201909:59:51GMT0800(中国标准时间)或者是一个数字:1486000791164写数字的日期,指定的毫秒数自1970年1月1日00:00:00到现在。1\.显示日期使用
Wesley13 Wesley13
2年前
java将前端的json数组字符串转换为列表
记录下在前端通过ajax提交了一个json数组的字符串,在后端如何转换为列表。前端数据转化与请求varcontracts{id:'1',name:'yanggb合同1'},{id:'2',name:'yanggb合同2'},{id:'3',name:'yang
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
皕杰报表(关于日期时间时分秒显示不出来)
在使用皕杰报表设计器时,数据据里面是日期型,但当你web预览时候,发现有日期时间类型的数据时分秒显示不出来,只有年月日能显示出来,时分秒显示为0:00:00。1.可以使用tochar解决,数据集用selecttochar(flowdate,"yyyyMMddHH:mm:ss")fromtablename2.也可以把数据库日期类型date改成timestamp
Wesley13 Wesley13
2年前
go中内存泄露的发现与排查
一,什么是内存泄漏Go中的并发性是以goroutine(独立活动)和channel(用于通信)的形式实现的。处理goroutine时,程序员需要小心翼翼地避免泄露。如果最终永远堵塞在I/O上(例如channel通信),或者陷入死循环,那么goroutine会发生泄露。即使是阻塞的goroutine,也会消耗资源
Stella981 Stella981
2年前
Spring RabbitMQ Channel理解
概述在AMQP协议中,有channel的概念,在RabbitMq中,channel表示逻辑连接或者叫虚拟连接,是棣属于TCP连接的。一个TCP连接里可以创建多个channel,在RabbitMQ里,消息的发送和接收都是基于channel的。!collection和channel的关系(htt
Wesley13 Wesley13
2年前
mysql中时间比较的实现
MySql中时间比较的实现unix\_timestamp()unix\_timestamp函数可以接受一个参数,也可以不使用参数。它的返回值是一个无符号的整数。不使用参数,它返回自1970年1月1日0时0分0秒到现在所经过的秒数,如果使用参数,参数的类型为时间类型或者时间类型的字符串表示,则是从1970010100:00:0
Stella981 Stella981
2年前
HIVE 时间操作函数
日期函数UNIX时间戳转日期函数: from\_unixtime语法:   from\_unixtime(bigint unixtime\, string format\)返回值: string说明: 转化UNIX时间戳(从19700101 00:00:00 UTC到指定时间的秒数)到当前时区的时间格式举例:hive   selec