3 回答

TA贡献1797条经验 获得超6个赞
这取决于您的程序,但一般来说我倾向于说不(您不需要在关闭频道之前清除频道):如果您关闭频道时频道中有项目,任何仍在从频道阅读的读者都会接收项目,直到通道为空。
下面是一个例子:
package main
import (
"sync"
"time"
)
func main() {
var ch = make(chan int, 5)
var wg sync.WaitGroup
wg.Add(1)
for range make([]struct{}, 2) {
go func() {
for i := range ch {
wg.Wait()
println(i)
}
}()
}
for i := 0; i < 5; i++ {
ch <- i
}
close(ch)
wg.Done()
time.Sleep(1 * time.Second)
}
在这里,程序将输出所有项目,尽管通道在任何读者甚至可以从通道读取之前严格关闭。

TA贡献1784条经验 获得超7个赞
有更好的方法来实现您想要实现的目标。您当前的方法只会导致丢弃一些记录,并随机处理其他记录(因为排水循环正在与所有消费者竞争)。这并没有真正解决目标。
你想要的是取消。这是Go 并发模式中的一个示例:管道和取消
func sq(done <-chan struct{}, in <-chan int) <-chan int {
out := make(chan int)
go func() {
defer close(out)
for n := range in {
select {
case out <- n * n:
case <-done:
return
}
}
}()
return out
}
您将一个done通道传递给所有 goroutine,并在您希望它们都停止处理时关闭它。如果你经常这样做,你可能会发现这个golang.org/x/net/context包很有用,它正式化了这个模式,并添加了一些额外的功能(比如超时)。

TA贡献1815条经验 获得超10个赞
我觉得所提供的答案实际上并没有澄清,除了不需要排水也不需要关闭的提示。因此,针对所描述的上下文的以下解决方案对我来说看起来很干净,它终止了工作人员并删除了对他们或相关频道的所有引用,因此,让 GC 清理频道及其内容:
type worker struct {
submitted chan Task
stop chan bool
p *Processor
}
// executed in a goroutine
func (w *worker) run() {
for {
select {
case task := <-w.submitted:
if err := task.Execute(w.p); err != nil {
logger.Error(err.Error())
}
case <-w.stop:
logger.Warn("Worker stopped")
return
}
}
}
func (p *Processor) Stop() {
if atomic.CompareAndSwapInt32(&p.status, running, stopped) {
for _, w := range p.workers {
w.stop <- true
}
// GC all workers as soon as goroutines stop
p.workers = nil
// GC all published data when workers terminate
p.submitted = nil
// no need to do the following above:
// close(p.submitted)
// for range p.submitted {}
}
}
- 3 回答
- 0 关注
- 142 浏览
添加回答
举报