go 从已关闭的channel读取数据

疑问1:从已关闭的channel读取数据会怎么样

刚接触go的channel,我在思考一个问题,chan是发送方关闭,还是接收方关闭,如果是发送方关闭,发送方发完数据,就关闭chan,那可能接收方还没接收完数据,读取关闭的信道会怎样?

验证1:有缓冲

package main

import (
	"fmt"
	"sync"
)

func main() {
	ch := make(chan int, 5)
	wg := sync.WaitGroup{}
	wg.Add(1)
	go func() {
		fmt.Println("wirte")
		for i := 1; i < 5; i++ {
			ch <- i
		}
		close(ch)
		fmt.Println("chan close")
		wg.Done()
	}()
	wg.Wait()
	fmt.Println("read")
	for i := 1; i < 9; i++ {
		v, ok := <-ch
		fmt.Println(v)
		fmt.Println(ok)
	}
}

结果

在这里插入图片描述
可以看到从已经关闭的chan是读取数据是没有问题的,但读取完chan的数据后再读取会返回false和默认值0

这样又产生一个新问题,接收方如何判断chan已经关闭了呢?
判断返回的是true还是false就可以知道,返回false代表已经关闭

验证2:无缓冲

package main

import (
	"fmt"
)

func main() {
	ch := make(chan int)
	go func() {
		fmt.Println("write")
		ch <- 1
		close(ch)
		fmt.Println("chan close")
	}()

	fmt.Println("read")
	for i := 0; i < 5; i++ {
		v, ok := <-ch
		fmt.Println(v)
		fmt.Println(ok)
	}
}

结果

在这里插入图片描述
结果和有缓冲一样

疑问2:重复关闭chan会怎么样

验证

package main

import (
	"fmt"
	"sync"
)

func main() {
	ch := make(chan int, 5)
	wg := sync.WaitGroup{}
	wg.Add(1)
	go func() {
		fmt.Println("wirte")
		for i := 1; i < 5; i++ {
			ch <- i
		}
		close(ch)
		close(ch)
		fmt.Println("chan close")
		wg.Done()
	}()
	wg.Wait()
	fmt.Println("read")
	for i := 1; i < 9; i++ {
		// v, ok := <-ch
		fmt.Println(<-ch)
		// fmt.Println(ok)
	}
}

结果

在这里插入图片描述重复关闭chan会panic

疑问3:向关闭的chan发送信号会怎样

验证

package main

import (
	"fmt"
	"sync"
)

func main() {
	ch := make(chan int, 5)
	wg := sync.WaitGroup{}
	wg.Add(1)
	go func() {
		fmt.Println("wirte")
		for i := 1; i < 5; i++ {
			ch <- i
		}
		close(ch)
		ch <- 5
		fmt.Println("chan close")
		wg.Done()
	}()
	wg.Wait()
	fmt.Println("stop")
}

结果

在这里插入图片描述

  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Go 语言Channel 在进行读、写、关闭等操作时会有一些特殊的行为。下面我将分别介绍在 nil、关闭channel、有数据的 channel 进行读、写、关闭时的行为。 1. 对于 nil 的 channel 进行读写操作: 对于 nil 的 channel 进行读写操作会导致 Goroutine 阻塞,并且会在运行时抛出 panic 异常。因此,在使用 channel 时,一定要确保它不是 nil。 2. 对于关闭channel 进行读写操作: 如果一个 channel 已经被关闭,那么对它进行写操作会导致 panic 异常。而对于读操作,如果 channel 还有数据,那么可以正常读取并返回数据。但如果 channel 已经没有数据了,那么读取操作会返回一个零,并且不会阻塞。因此,在使用 channel 时,一定要注意判断 channel 是否已经被关闭。 3. 对于有数据的 channel 进行读写操作: 对于有数据的 channel 进行写操作会阻塞当前 Goroutine,直到有其他 Goroutine 读取了数据。而对于读操作,如果 channel 还有数据,那么可以正常读取并返回数据。但如果 channel 已经没有数据了,那么读取操作会阻塞当前 Goroutine,直到有其他 Goroutine 写入了数据或者 channel关闭。如果 channel关闭了,那么读取操作会返回一个零,并且不会阻塞。 总之,当对 channel 进行读写操作时,需要注意它的状态,避免出现不必要的阻塞或异常。同时,对于已经关闭channel,不应该再进行写操作,否则会导致 panic 异常。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值