go - 处理超时和收听频道
问题描述
我有看起来像这样的代码,我正在收听一个频道,直到超时间隔。让我们说这个 goroutine1
select {
case <-time.After(TimeoutInterval):
mu.Lock()
defer mu.Unlock()
delete(msgChMap, index)
return ""
case msg := <-msgCh:
return msg
}
在其他地方,我有一个2
运行类似这样的 goroutine,它msgCh
从 Map 中获取适当的内容,删除 map 中的条目,然后通过通道发送消息。
mu.Lock()
msgCh, ok := msgChMap[index]
delete(msgChMap, index)
mu.Unlock()
if ok {
msgCh <- "yay"
}
似乎我可以msgCh
从 Map 中获取消息通道,尝试发送消息,但是因为TimeoutInterval
已经过去,所以不会有任何东西在听该通道,并且我的代码会卡住等待侦听器。如果我在发送yay
到后加锁msgCh
,我可能会死锁,因为2
将等待通道的侦听器并且不释放锁,但1
不再侦听但需要锁。
避免等待听众卡住的一般模式是什么?也许 go 足够聪明,不会被困在这里。
解决方案
这里的问题是频道阅读器在作者不知道的情况下停止了。应该可以构造这个解决方案,使这种情况永远不会发生,但是暂时忽略这一点,对于这个特定问题,您需要的是对通道本身的原子访问,以及通道状态的标志:
type channel struct {
sync.Mutex
msgCh chan Msg
active bool
}
现在通过锁定通道来完成写入通道:
ch.Lock()
if ch.active {
ch.msgCh<-data
}
ch.Unlock()
当您“停用”频道时,请重置标志:
case <-time.After(TimeoutInterval):
mu.Lock()
defer mu.Unlock()
ch.Lock()
defer ch.Unlock()
delete(msgChMap, index)
ch.active=false
return ""
当然,现在你必须*channel
在你的地图中保留一个。
推荐阅读
- sql-server - SQL Server 中 varchar(n) 和 varchar(m) 之间的效率差异——这与 varchar(n) 和 varchar(MAX) 无关
- node.js - Docker 镜像构建卡在 ReactJS 的“npm install”
- envoyproxy - Envoy Sidecar 代理会话持久化?
- vb.net - 如何从 vb.net 中的调用对象访问数据
- matlab - 如何在 Octave 中转换 Matlab 的 Tiff 函数?
- r - 如何捕获所有成功的会话?
- apache-spark - Common Crawl:pyspark,无法使用
- reactjs - 使用 Apollo Provider 的钩子
- javascript - 我可以在箭头函数中调用这个函数吗?
- ios - 如何在 SwiftUI 中删除重复的集合元素