go channel中的 close注意事项 range取数据

在使用 Go 语言中的 `close` 函数时,有一些注意事项需要牢记,以确保程序的健壮性和正确性:

  1. **仅用于通道(channel)**:
  • `close` 函数只能用于关闭通道,不能用于关闭文件、网络连接或其他资源。
  1. **关闭的通道不能再发送数据**:
  • 一旦通道被关闭,不能再向该通道发送数据,否则会导致运行时恐慌(panic)。
  1. **关闭的通道仍然可以接收数据**:
  • 从已关闭的通道接收数据是安全的。如果通道中有缓冲数据,接收操作将继续直到缓冲区被耗尽。之后,接收操作将返回零值。
  1. **避免重复关闭**:
  • 试图关闭已经关闭的通道会导致运行时恐慌(panic)。因此,确保通道只被关闭一次。
  1. **关闭责任**:
  • 通道的发送方通常负责关闭通道,而不是接收方。
  1. **单向通道**:
  • 不能关闭一个只接收(receive-only)的通道,因为它无法发送数据。
  1. **使用 `ok`-idiom 检查通道状态**:
  • 在接收数据时,可以使用 `v, ok := <-ch` 语法检查通道是否已关闭。`ok` 为 `false` 表示通道已关闭。

以下是一个简单的示例,展示如何正确使用 `close`:

package main

import (

"fmt"

)

func main() {

ch := make(chan int, 2)

// 发送数据

ch <- 1

ch <- 2

// 关闭通道

close(ch)

// 接收数据

for v := range ch {

fmt.Println(v)

}

// 检查关闭状态

if _, ok := <-ch; !ok {

fmt.Println("通道已关闭")

}

}

通过遵循这些注意事项,可以有效地避免由于不正确使用 `close` 而引起的错误。

使用 range 迭代通道。

当你对一个通道使用 range 时,它会不断从通道中接收值,直到通道被关闭并且通道中的所有值都被接收完毕。

package main

import (

"fmt"

)

func main() {

ch := make(chan int, 3)

// 向通道发送数据

go func() {

for i := 1; i <= 3; i++ {

ch <- i

}

close(ch) // 关闭通道

}()

// 使用 range 迭代通道

for value := range ch {

fmt.Println(value)

}

fmt.Println("通道已关闭,迭代结束")

}

相关推荐
程序员小假1 分钟前
我们来说一说 悲观锁、乐观锁、分布式锁的使用场景和使用技巧
后端
_祝你今天愉快3 分钟前
Java Lock
android·java·后端
listhi5204 分钟前
Python实现信号小波分解与重构
开发语言·python·重构
jzy371113 分钟前
minio集群安装(3节点模拟4节点)
后端
林太白24 分钟前
Rust新增优化
后端·rust
熊猫片沃子35 分钟前
mybatis 与mybatisplus 比较总结
java·后端·mybatis
骑驴看星星a38 分钟前
层次分析法代码笔记
开发语言·笔记·python·numpy
brzhang1 小时前
昨天我和同事聊聊架构这事儿,特别是怎么才能睡个好觉,有点点收获
前端·后端·架构
风象南1 小时前
告别YAML,在SpringBoot中用数据库配置替代配置文件
spring boot·后端
brzhang1 小时前
OpenAI 终究还是背刺了自己:1200亿参数模型直接开源,实测 120b 模型编码能力强过 Claude3.5!
前端·后端·架构