ZBLOG

time.Tick 每周10点触发一次

在 Go 中,time.Tick 主要用于 固定时间间隔(如每秒/每分钟)的触发,但无法直接实现 每周特定时间点(如每周一 10:00)的触发。以下是两种实现方式:


方案1:基于 time.Tick + 时间判断

通过每秒检查当前时间是否匹配目标时间点:

package main

import (
	"fmt"
	"time"
)

func weeklyTick(targetWeekday time.Weekday, targetHour int) {
	for now := range time.Tick(1 * time.Second) { // 每秒检查一次
		if now.Weekday() == targetWeekday && now.Hour() == targetHour && now.Minute() == 0 && now.Second() == 0 {
			fmt.Printf("Triggered at: %v\n", now)
			// 执行你的任务...
		}
	}
}

func main() {
	go weeklyTick(time.Monday, 10) // 每周一10:00:00触发
	select {} // 阻塞主 Goroutine
}

缺点:每秒轮询,不够高效。


方案2:精确计算下次触发时间(推荐)

动态计算距离下次目标时间的间隔,避免轮询:

package main

import (
	"fmt"
	"time"
)

func nextWeeklyTick(targetWeekday time.Weekday, targetHour int) time.Duration {
	now := time.Now()
	
    // 计算下一个目标时间点
	next := time.Date(
        now.Year(), now.Month(), now.Day(), 
        targetHour, 0, 0, 0, 
        now.Location(),
    )
    
    // 如果本周已过目标时间,跳到下周
	for next.Before(now) || next.Weekday() != targetWeekday {
        next = next.AddDate(0, 0, 1)
    }
    
	return next.Sub(now)
}

func weeklyTicker(targetWeekday time.Weekday, targetHour int) {
	for {
        waitDuration := nextWeeklyTick(targetWeekday, targetHour)
        <-time.After(waitDuration)
        
        fmt.Printf("Triggered at: %v\n", time.Now())
        // 执行你的任务...
    }
}

func main() {
	go weeklyTicker(time.Monday, 10) // 每周一10:00:00触发
	select {}
}

优点:无轮询,精准高效。


关键点

  1. 时区处理:确保 time.Location() (如 time.Local)与你的业务时区一致。
  2. 容错性:若任务执行时间长于间隔,需考虑并发控制(如加锁)。
  3. 退出机制:可通过 context.Context 实现优雅退出。

如果需要更复杂的调度(如支持 Cron 表达式),建议使用第三方库如 github.com/robfig/cron


内容由零声教学AI助手提供,问题来源于学员提问

本站部分文章来源于网络,版权归原作者所有,如有侵权请联系站长删除。
转载请注明出处:https://golang.0voice.com/?id=24058

分享:
扫描分享到社交APP
上一篇
下一篇
发表列表
游客 游客
此处应有掌声~
评论列表

还没有评论,快来说点什么吧~

联系我们

在线咨询: 点击这里给我发消息

微信号:3007537140

上班时间: 10:30-22:30

关注我们
x

注册

已经有帐号?