登录
首页 >  Golang >  Go问答

使用Golang的time.Ticker实现滴答时钟

来源:stackoverflow

时间:2024-02-15 21:15:22 152浏览 收藏

珍惜时间,勤奋学习!今天给大家带来《使用Golang的time.Ticker实现滴答时钟》,正文内容主要涉及到等等,如果你正在学习Golang,或者是对Golang有疑问,欢迎大家关注我!后面我会持续更新相关内容的,希望都能帮到正在学习的大家!

问题内容

我正在开发一个 go 程序,它要求我在(相当)精确的时钟时间运行某些函数(例如,每 5 分钟一次,但具体在 3:00、3:05、3:10 等) ,而不仅仅是程序开始后每 5 分钟一次)。

在来这里请求你的帮助之前,我尝试实现一个自动收报机来做到这一点,尽管它看起来工作正常,但感觉有点肮脏/老套,而且不是超级精确(它只有几分之一毫秒的误差,但我想知道是否有理由相信差异会随着时间的推移而增加)。

我当前的实现如下,我真正要问的是,是否有更好的解决方案来实现我想要实现的目标(并且我可以更有信心)?

type ScheduledTicker struct {
    C chan time.Time
}

// NewScheduledTicker returns a ticker that ticks on defined intervals after the hour
// For example, a ticker with an interval of 5 minutes and an offset of 0 will tick at 0:00:00, 0:05:00 ... 23:55:00
// Using the same interval, but an offset of 2 minutes will tick at 0:02:00, 0:07:00 ... 23:57
func NewScheduledTicker(interval time.Duration, offset time.Duration) *ScheduledTicker {
    s := &ScheduledTicker{
        C: make(chan time.Time),
    }

    go func() {
        now := time.Now()

        // Figure out when the first tick should happen
        firstTick := now.Truncate(interval).Add(interval).Add(offset)

        // Block until the first tick
        <-time.After(firstTick.Sub(now))

        t := time.NewTicker(interval)

        // Send initial tick
        s.C <- firstTick

        for {
            // Forward ticks from the native time.Ticker to the ScheduledTicker channel
            s.C <- <-t.C
        }
    }()

    return s
}

正确答案


所有平台上的大多数计时器 api 都按照系统时间而不是挂钟时间工作。您所表达的是有一个挂钟间隔。

正如另一个答案所表达的,有可用的开源软件包。在 google 上快速搜索“golang wall clockticker”会产生有趣的结果。

还有一件事要考虑。在 windows 上有“计划任务”,在 linux 上有“cronjobs”,可以为您执行挂钟唤醒间隔。如果您的程序要做的只是在完成所需工作之前在所需间隔之间睡眠/滴答,请考虑使用它。

但是如果你自己构建它......

尝试在挂钟间隔内完成工作会变得很复杂,因为当笔记本电脑盖关闭时台式电脑会进入睡眠状态(暂停系统时间)以及系统和挂钟之间的时钟偏差。有时用户喜欢更改 pc 的时钟 - 您可以醒来并轮询时间。现在,您会发现自己处于昨天!这在云中运行的服务器上不太可能发生,但在个人设备上却真实存在。

在我的产品团队中,当我们确实需要时钟时间或需要以超过一个小时的时间间隔做某事时,我们会以更频繁的时间间隔醒来,看看“时间到了”。例如,如果我们想要每 12 小时执行一次,我们可能会每小时醒来并轮询时间。 (我工作的地方我们使用 c++,而不是 go)。

否则,我的 5 分钟间隔的一般算法是睡眠(或设置系统计时器)1 分钟或更短。每次从 time.sleep 返回后,检查当前时间 (time.now()) 以查看当前时间是否等于或晚于下一个预期间隔时间。发布您的频道事件,然后重新计算下一次唤醒时间。如果您虚假地早起,您甚至可以更改睡眠时间的粒度。

但是要小心! golang time 对象包含挂钟和系统时钟时间。这包括 time.now() 返回的结果。 time.add()time.sub() 和其他一些时间比较函数首先在 monotonic time 上工作,然后再转到挂钟时间。您可以通过执行以下操作从 time.now 结果中去除单调时间:

func GetWallclockNow() time.Time {
    var t time.Time = time.Now()
    return time.Date(t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), t.Second(), t.Nanosecond(), t.Location())
}

然后像 addafter 这样的后续操作将在挂钟空间中。

到这里,我们也就讲完了《使用Golang的time.Ticker实现滴答时钟》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>