
深入go语言time.afterfunc函数:避免阻塞与正确用法
Go语言time.AfterFunc函数用于在指定时间后执行某个函数。本文将探讨其正确用法,并解决一个常见的误用场景。
示例代码旨在3秒后打印"done",随后打印"ok"。然而,错误地尝试通过监听time.AfterFunc返回的Timer的C通道来同步,导致错误。这是因为time.AfterFunc返回的Timer的C通道始终为nil,无法监听。
错误示例(会报错): 该部分省略,因为原文中未提供完整的错误示例代码。
正确用法:使用通道同步
立即学习“go语言免费学习笔记(深入)”;
为了实现先打印"done",再打印"ok"的效果,且避免使用time.Sleep,我们可以利用Go语言的通道机制进行同步。
改进后的代码如下:
package main
import (
"fmt"
"time"
)
func main() {
c := make(chan struct{}) // 使用无缓冲通道
time.AfterFunc(3*time.Second, func() {
fmt.Println("done")
close(c) // 关闭通道,通知主goroutine
})
<-c // 阻塞等待通道关闭
fmt.Println("ok")
}
在这个改进版本中:
- 我们创建了一个无缓冲通道
c,用于在time.AfterFunc的匿名函数执行完毕后进行同步。 - 在匿名函数中,打印"done"后,我们关闭通道
c,向主goroutine发出信号。 - 主goroutine通过
阻塞等待通道c关闭,确保"done"打印完成后再打印"ok"。
此方法有效避免了阻塞,并清晰地实现了预期的执行顺序。 记住,time.AfterFunc本身是非阻塞的;同步控制需要通过其他机制实现,例如通道。










