2 回答
TA贡献1873条经验 获得超9个赞
修改中断处理以在中断时关闭通道。这允许多个 goroutines 通过等待通道关闭来等待事件。
shutdown := make(chan struct{})
interrupt := make(chan os.Signal, 1)
signal.Notify(interrupt, os.Interrupt)
go func() {
<-interrupt
log.Println("interrupt")
close(shutdown)
}()
将每个连接代码移动到一个函数中。这段代码是从问题中复制粘贴的,有两个变化:中断通道被关闭通道替换;该函数在函数完成时通知 sync.WaitGroup。
func connect(u string, shutdown chan struct{}, wg *sync.WaitGroup) {
defer wg.Done()
log.Printf("connecting to %s", u)
c, _, err := websocket.DefaultDialer.Dial(u, nil)
if err != nil {
log.Fatal("dial:", err)
}
defer c.Close()
done := make(chan struct{})
go func() {
defer close(done)
for {
_, message, err := c.ReadMessage()
if err != nil {
log.Println("read:", err)
return
}
log.Printf("recv: %s", message)
}
}()
ticker := time.NewTicker(time.Second)
defer ticker.Stop()
for {
select {
case <-done:
return
case t := <-ticker.C:
err := c.WriteMessage(websocket.TextMessage, []byte(t.String()))
if err != nil {
log.Println("write:", err)
return
}
case <-shutdown:
// Cleanly close the connection by sending a close message and then
// waiting (with timeout) for the server to close the connection.
err := c.WriteMessage(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseNormalClosure, ""))
if err != nil {
log.Println("write close:", err)
return
}
select {
case <-done:
case <-time.After(time.Second):
}
return
}
}
}
在中声明一个sync.WaitGroupmain()
。对于要连接到的每个 websocket 端点,递增 WaitGroup 并启动 goroutine 以连接该端点。启动 goroutine 后,在 WaitGroup 上等待 goroutine 完成。
var wg sync.WaitGroup
for _, u := range endpoints { // endpoints is []string
// where elements are URLs
// of endpoints to connect to.
wg.Add(1)
go connect(u, shutdown, &wg)
}
wg.Wait()
TA贡献1786条经验 获得超13个赞
与每个不同服务器的通信是否完全独立于其他服务器?如果是的话,我会以这样的方式四处走动:
在main中创建一个带有取消函数的上下文
在 main 中创建一个等待组来跟踪启动的 goroutines
对于每个服务器,添加到等待组,从传递上下文和等待组引用的主函数启动一个新的 goroutine
main进入一个 for/select 循环,监听信号,如果信号到达,调用 cancelfunc 并等待等待组。
main还可以监听来自 goroutines 的结果 chan,并可能自己打印结果,因为 goroutines 不应该直接这样做。
正如我们所说,每个goroutine都有 wg 的引用、上下文和可能的 chan 以返回结果。现在,如果 goroutine 必须只做一件事,或者它是否需要做一系列事情,这个方法就会分裂。对于第一种方法
如果只有一件事要做,我们会遵循此处描述的方法(观察到异步他会依次启动一个新的 goroutine 来执行 DoSomething() 步骤,该步骤将在通道上返回结果)这允许它能够随时接受取消信号。由您决定您想要的非阻塞程度以及您想要响应取消信号的速度。此外,将关联的上下文传递给 goroutines 的好处是您可以调用 Context enabled大多数库函数的版本。例如,如果您希望您的拨号超时为 1 分钟,您可以创建一个新的上下文,该上下文从传递的超时开始,然后使用该超时创建 DialContext。这允许拨号从超时或父(您在 main 中创建的)上下文的 cancelfunc 被调用时停止。
如果需要做更多的事情,我通常更喜欢用 goroutine 做一件事,让它调用一个新的 goroutine 来执行下一步(将所有引用传递到管道中)然后退出。
这种方法可以很好地扩展取消,并能够在任何步骤停止流水线,并且可以轻松地为可能需要太长时间的步骤支持带有 dealines 的上下文。
- 2 回答
- 0 关注
- 232 浏览
添加回答
举报