栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 面试经验 > 面试问答

长度未知的关闭通道

面试问答 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

长度未知的关闭通道

通道关闭后,您将无法在该通道上发送更多值,否则会出现混乱。这就是您的经验。

这是因为您启动了使用同一个通道的多个goroutine,并且它们在该通道上发送值。然后关闭每个通道。而且由于它们未同步,因此一旦第一个goroutine到达将其关闭的点,其他人可能(并且他们将)仍然继续在其上发送值:恐慌!

您只能关闭该通道一次(尝试关闭一个已经关闭的通道也会引起混乱)。当所有发送值的goroutine完成后,您应该这样做。为此,您需要检测何时完成所有发送方goroutine。一种检测到此情况的惯用方法是使用

sync.WaitGroup

对于每个启动的发送方goroutine,我们在

WaitGroup
using中添加1
WaitGroup.Add()
。每个完成发送值的goroutine都可以通过调用发出信号
WaitGroup.Done()
。最好以延迟语句的方式执行此操作,因此,如果您的goroutine会突然终止(例如,紧急情况),
WaitGroup.Done()
仍会被调用,并且不会使其他goroutine挂起(等待赦免-
WaitGroup.Done()
永远不会出现的“缺失” 调用)。 )。

并且

WaitGroup.Wait()
将等待,直到所有发送方goroutine都完成,并且只有在此之后,才关闭通道一次。我们希望检测到此“全局”完成事件并在处理正在进行的发送值时关闭通道,因此我们必须在自己的goroutine中执行此操作。

因为我们

for ...range
在通道上使用了构造,所以接收器goroutine将一直运行到通道关闭为止。并且由于它在主goroutine中运行,因此直到从通道正确接收并处理了所有值后,程序才会退出。所述
for... range
构建体循环,直到接收到所有的值的信道被关闭之前已被发送。

请注意,下面的解决方案也适用于缓冲通道和非缓冲通道,而无需进行修改(请尝试使用带有的缓冲通道

ch := make(chan int, 100)
)。

正确的解决方案(在Go Playground上尝试):

func gen(ch chan int, wg *sync.WaitGroup) {    defer wg.Done()    var i int    for {        time.Sleep(time.Millisecond * 10)        ch <- i        i++        // when no more data (e.g. from db, or event stream)        if i > 100 { break        }    }}func receiver(ch chan int) {    for i := range ch {        fmt.Println("received:", i)    }}func main() {    ch := make(chan int)    wg := &sync.WaitGroup{}    for i := 0; i < 10; i++ {        wg.Add(1)        go gen(ch, wg)    }    go func() {        wg.Wait()        close(ch)    }()    receiver(ch)}

注意:

请注意,

receiver(ch)
在主goroutine
中运行非常重要,而代码
WaitGroup
在其自身的(非主)goroutine中等待和关闭通道的代码;而不是相反。如果您将它们切换为2,则可能会导致“提前退出”,即并非所有值都可以从通道中接收和处理。这样做的原因是因为Go程序在主goroutine完成时退出(请参阅:程序执行)。它不等待其他(非主)goroutine完成。因此,如果等待和关闭通道位于主goroutine中,则在关闭通道之后,程序可以随时退出,而无需等待其他goroutine,在这种情况下,该goroutine将循环从该通道接收值。



转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/469004.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号