Go定时提醒应用核心是time.AfterFunc(单次提醒)和time.Ticker(周期提醒),结合time.Parse解析时间、channel控制生命周期,需注意资源释放与goroutine管理。
用 Go 实现定时提醒应用,核心是 time 包中的 time.Ticker 和 time.AfterFunc,它们适合不同场景:周期性提醒用 Ticker,单次延时提醒用 AfterFunc。关键不在“轮询”或“死循环”,而在于利用 Go 的并发模型和通道机制优雅地触发事件。
适合设置一个未来时间点触发一次通知(比如 5 分钟后提醒喝水)。
sync.WaitGroup 或 chan struct{}
func main() {
duration := 3 * time.Second
fmt.Println("将在", duration, "后提醒...")
time.AfterFunc(duration, func() {
fmt.Println("⏰ 提醒:该休息一下啦!")
})
// 防止主 goroutine 立即退出
time.Sleep(4 * time.Second)
}适合每分钟检查日程、每小时发送状态报告等固定间隔任务。
time.NewTicker 返回一个 *Ticker,其 C 字段是 ,每次到达间隔就发送当前时间
ticker.Stop() 避免资源泄漏;通常配合 select + case 使用
func main() {
ticker := time.NewTicker(2 * time.Second)
defer ticker.Stop()
done := make(chan bool)
go func() {
for i := 0; i < 3; i++ {
<-ticker.C
fmt.Printf("? 第 %d 次提醒\n", i+1)
}
done <- true
}()
<-done // 等待完成}
真实提醒应用需要支持“明天 9:30”、“每周一 8:00”这类自然表达,但 time 包本身不解析字符串,需手动处理基础格式。
time.Parse 配合固定 layout(如 "2006-01-02 15:04")转换字符串为 time.Time
time.Now().Add(...) 构造未来时刻,再传给 AfterFunc
layout := "2006-01-02 15:04"
now := time.Now()
todayStr := now.Format("2006-01-02")
targetStr := todayStr + " 14:30"
targetTime, err := time.Parse(layout, targetStr)
if err != nil {
log.Fatal(err)
}
if targetTime.Before(now) {
targetTime = targetTime.Add(24 * time.Hour) // 推到明天
}
delay := targetTime.Sub(now)
time.AfterFunc(delay, func() {
fmt.Println("⏰ 到了设定的提醒时间!")
})
实际应用中,用户可能随时取消或暂停提醒。纯靠 Ticker.Stop() 不够灵活,可引入控制通道。
select
监听 ticker.C 和自定义控制信号(如 pauseCh, stopCh)AfterFunc 无法取消,建议改用 time.After + select 超时判断)time.After + select,便于通过 close(cancelCh) 中断等待不复杂但容易忽略的是时间精度与 goroutine 生命周期管理。只要记住:用 AfterFunc 做一次性动作,用 Ticker 做周期动作,用 select + channel 做流程控制,再辅以合理的错误处理和资源清理,就能写出健壮的 Go 提醒服务。