
本文深入探讨了 Golang 中 time.Ticker 的停止行为。Ticker.Stop() 仅停止计时器,并不会关闭通道,导致在 range 循环中监听通道的 Goroutine 永久阻塞。本文将提供一种使用额外通道来优雅地停止 Ticker 并退出 Goroutine 的方法,确保资源得到正确释放。
在 Golang 中,time.Ticker 用于周期性地发送时间事件到其通道 C。然而,当使用 Ticker.Stop() 停止计时器时,需要特别注意其行为。Stop() 方法会停止计时器,但不会关闭通道 C。这意味着如果有一个 Goroutine 正在使用 range 循环监听 ticker.C,它将永远阻塞,因为通道永远不会关闭。
考虑以下示例:
package main
import (
"log"
"time"
)
func main() {
ticker := time.NewTicker(1 * time.Second)
go func() {
for _ = range ticker.C {
log.Println("tick")
}
log.Println("stopped")
}()
time.Sleep(3 * time.Second)
log.Println("stopping ticker")
ticker.Stop()
time.Sleep(3 * time.Second)
}运行此代码,会发现 Goroutine 在调用 ticker.Stop() 后并没有退出,"stopped" 消息永远不会打印。这是因为 range ticker.C 会一直等待通道接收数据,而 Stop() 只是停止了计时器发送数据,并没有关闭通道。
立即学习“go语言免费学习笔记(深入)”;
为了解决这个问题,可以使用一个额外的通道来通知 Goroutine 退出循环。以下是一个改进后的示例:
package main
import (
"log"
"time"
)
// Run the function every tick
// Return false from the func to stop the ticker
func Every(duration time.Duration, work func(time.Time) bool) chan bool {
ticker := time.NewTicker(duration)
stop := make(chan bool, 1)
go func() {
defer log.Println("ticker stopped")
for {
select {
case time := <-ticker.C:
if !work(time) {
stop <- true
}
case <-stop:
ticker.Stop() // Important: Stop the ticker before exiting
return
}
}
}()
return stop
}
func main() {
stop := Every(1*time.Second, func(time.Time) bool {
log.Println("tick")
return true
})
time.Sleep(3 * time.Second)
log.Println("stopping ticker")
stop <- true
time.Sleep(3 * time.Second)
}在这个解决方案中:
正确处理 Golang 中 time.Ticker 的停止行为至关重要,可以避免 Goroutine 永久阻塞和资源泄露。 通过使用额外的通道来发送停止信号,并确保在 Goroutine 退出前调用 ticker.Stop(),可以优雅地停止计时器并释放相关资源。 本文提供的示例代码和注意事项可以帮助你更好地理解和应用 time.Ticker。
以上就是Golang 中 Ticker 的停止行为详解及正确处理方式的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号