

新闻资讯
技术学院用time.Ticker适合轻量周期任务,但需goroutine避免阻塞;robfig/cron/v3支持cron表达式和秒级调度,需显式启用秒级并自行recover;自研调度器易引发并发、内存泄漏等问题;跨机调度须用分布式锁或消息队列。
time.Ticker 实现简单周期任务,但别直接上生产如果只是每 5 秒打印一次日志、或轮询某个本地状态,time.Ticker 足够轻量且可控。它不依赖外部服务,启动快,无额外依赖。
但要注意:time.Ticker 是纯内存级调度,进程退出即失效;不支持任务持久化、失败重试、分布式协调。一旦程序 panic 或被 kill,任务就彻底消失。
常见误用是把耗时操作(如 HTTP 请求、数据库写入)直接塞进 ticker.C 的 for 循环里,导致下一轮 tick 被阻塞——必须用 goroutine 包裹:
ticker := time.NewTicker(5 * time.Second) defer ticker.Stop()for { select { case <-ticker.C: go func() { // 这里放实际任务逻辑 doSomething() }() } }
robfig/cron/v3 处理 cron 表达式与单机多任务管理需要按“每天凌晨2点”“每周一上午9点”这类语义调度?robfig/cron/v3 是当前 Go 生态最稳定的 cron 库,支持标准 cron 格式、秒级精度(加前导字段)、任务名称标记和基础运行统计。
关键点:
cron.New(cron.WithSeconds()) 必须显式启用秒级支持,否则默认从分钟开始解析cron.AddFunc() 注册的任务,底层共享同一个系统 timer,高频率任务(如 * * * * * *)可能挤压其他任务执行时机cron.Stop() + cron.Start() 重建,期间有短暂窗口丢失触发看似灵活,实则埋坑密集:
sync.RWMutex,
但锁粒度难把握——锁太粗影响吞吐,太细易漏保护context.WithTimeout)、取消(ctx.Done())、重试策略已有成熟库(如 gocron、asynq)已覆盖这些细节,重复造轮子成本远高于学习接入成本。
单机 cron 在多实例部署时会重复执行同一任务(比如两个 API 实例都触发“每日数据归档”),这不是 bug,是设计使然。
解法只有两类:
SET key value NX PX 10000)包装任务入口,抢到锁的实例才执行——适合低频、非强一致场景asynq 或自研)按计划推送到 Redis / Kafka,各工作节点消费执行——适合高可靠、可追溯、需重试的业务注意:不要用本地文件或数据库行锁做分布式协调,网络分区或事务异常时极易出现脑裂或死锁。
真正麻烦的不是“怎么让任务跑起来”,而是“怎么确保它只跑一次、失败能恢复、变更可灰度、执行可审计”。这些边界问题,往往在压测或上线后才暴露。