• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    Golang优雅关闭channel的方法示例

    前言

    最近使用go开发后端服务,服务关闭需要保证channel中的数据都被读取完,理由很简单,在收到系统的中断信号后,系统需要做收尾工作,保证channel的数据都要被处理掉,然后才可以关闭系统。但实现起来没那么简单,下面来一起看看详细的介绍吧。

    关于Go channel设计和规范的批评:

    所以Golang 内建的 close 方法可以关闭 channel,如果往已经关闭的 channel 发送数据,则会报错:panic: close of closed channel.

    看如下代码,在一段时间内,生产者可以不断往 channel 写入数据,消费者进行处理,一段时间后 channel 关闭了,这个时候如果还有数据往 channel 发送,程序就会报错。

    package main
     
    import (
     "fmt"
     "sync"
     "time"
    )
     
    func main() {
     jobs := make(chan int)
     var wg sync.WaitGroup
     go func() {
     time.Sleep(time.Second * 3)
     close(jobs)
     }()
     go func() {
     for i := 0; ; i++ {
     jobs - i
     fmt.Println("produce:", i)
     }
     }()
     wg.Add(1)
     go func() {
     defer wg.Done()
     for i := range jobs {
     fmt.Println("consume:", i)
     }
     }()
     wg.Wait()
    }

    多运行几次出错的概率会比较大:

    produce: 33334
    consume: 33334
    consume: 33335
    produce: 33335
    produce: 33336
    consume: 33336
    consume: 33337
    produce: 33337
    produce: 33338
    consume: 33338
    consume: 33339
    produce: 33339
    produce: 33340
    consume: 33340
    panic: send on closed channel
     
    goroutine 19 [running]:
    panic(0x49b660, 0xc042410bb0)
      C:/Go/src/runtime/panic.go:500 +0x1af
    main.main.func2(0xc04203a180)
      C:/Users/tanteng/Go/src/examples/channel_close.go:18 +0x6b
    created by main.main
      C:/Users/tanteng/Go/src/examples/channel_close.go:21 +0xb8
    exit status 2

    如何优雅关闭 channel

    那么在往通道发数据前如何判断通道是否关闭呢?

    1._,ok := - jobs

    此时如果 channel 关闭,ok 值为 false,如果 channel 没有关闭,则会漏掉一个 jobs

    2.使用 select 方式

    再创建一个 channel,叫做 timeout,如果超时往这个 channel 发送 true,在生产者发送数据给 jobs 的 channel,用 select 监听 timeout,如果超时则关闭 jobs 的 channel.

    完整代码如下:

    package main
     
    import (
     "fmt"
     "sync"
     "time"
    )
     
    func main() {
     jobs := make(chan int)
     timeout := make(chan bool)
     var wg sync.WaitGroup
     go func() {
     time.Sleep(time.Second * 3)
     timeout - true
     }()
     go func() {
     for i := 0; ; i++ {
     select {
     case -timeout:
     close(jobs)
     return
     
     default:
     jobs - i
     fmt.Println("produce:", i)
     }
     }
     }()
     wg.Add(1)
     go func() {
     defer wg.Done()
     for i := range jobs {
     fmt.Println("consume:", i)
     }
     }()
     wg.Wait()
    }

    这样就可以保证不会往已经关闭的 channel 中发送数据了。

    总结

    以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

    您可能感兴趣的文章:
    • 基于golang channel实现的轻量级异步任务分发器示例代码
    • golang中for循环遍历channel时需要注意的问题详解
    • golang实现基于channel的通用连接池详解
    • golang中单向channel的语法介绍
    • golang判断chan channel是否关闭的方法
    • Golang中channel使用的一些小技巧
    • Golang中channel的原理解读(推荐)
    上一篇:Go语言实现遗传算法的实例代码
    下一篇:Golang实现for循环运行超时后自动退出的方法
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    Golang优雅关闭channel的方法示例 Golang,优雅,关闭,channel,的,