如何使用Golang ticker实现定时任务并发执行_周期性触发协程

发布时间 - 2026-01-07 00:00:00    点击率:
Go 中用 ticker 实现周期性协程需防堆积、泄漏和竞态:Ticker 仅发信号,任务需手动控制并发;可用信号量限流、atomic.Bool 防重入;必须调用 Stop() 并结合 context 优雅退出。

Go 语言中用 ticker 实现周期性触发协程并不难,关键在于避免协程堆积、资源泄漏和竞态问题。Ticker 本身只负责“准时发信号”,真正执行任务的逻辑必须主动控制并发行为。

理解 Ticker 的基本用法

Ticker 是一个按固定间隔发送时间戳的通道,它不会自动执行任何函数。你得手动从 ticker.C 接收信号,再启动协程处理任务:

ticker := time.NewTicker(5 * time.Second)
defer ticker.Stop()

for range ticker.C { go func() { // 执行任务 doWork() }() }

但这样写有隐患:如果 doWork() 执行时间超过 5 秒,每次 tick 都会新启一个 goroutine,导致协程无限堆积。

限制并发数:用带缓冲的 WaitGroup 或信号量

要防止协程泛滥,需对同时运行的任务数量做限制。推荐用 semaphore(信号量)或带缓冲 channel 模拟:

  • 定义一个容量为 N 的 channel,每次任务开始前尝试获取一个“令牌”
  • 任务结束时归还令牌,确保最多只有 N 个任务并发执行
  • 若令牌已满,可选择跳过本次 tick、排队等待,或直接丢弃
sem := make(chan struct{}, 3) // 最多 3 个并发

for range ticker.C { select { case sem <- struct{}{}: go func() { defer func() { <-sem }() // 归还令牌 doWork() }() default: // 令牌不足,跳过本次执行(也可记录日志) log.Println("skipped: too many tasks running") } }

防重入:避免同一时刻多个相同任务并行

有些任务不允许并发执行(比如写配置文件、清理临时目录)。这时可在任务外加锁,或用原子状态标记:

  • sync.Mutex 包裹任务入口,确保同一时间只有一个实例在跑
  • 更轻量的方式是用 atomic.Bool 标记“是否正在运行”,tick 触发时先 CAS 尝试置为 true,失败则跳过
var running atomic.Bool

for range ticker.C { if !running.CompareAndSwap(false, true) { log.Println("task already running, skip") continue }

go func() {
    defer running.Store(false)
    doWork()
}()

}

优雅退出与资源回收

程序退出前必须调用 ticker.Stop(),否则 ticker 会持续向 channel 发送时间,造成 goroutine 泄漏。建议配合 context 管理生命周期:

  • context.WithCancel 创建可取消上下文
  • 在主 goroutine 监听 cancel 信号,触发后 stop ticker 并等待所有任务完成
  • 任务内部也应监听 ctx.Done(),及时中断耗时操作
ctx, cancel := context.WithCancel(context.Background())
ticker := time.NewTicker(5 * time.Second)
defer func() {
    ticker.Stop()
    cancel()
}()

go func() { for { select { case <-ticker.C: go doWorkWithContext(ctx) case <-ctx.Done(): return } } }()

不复杂但容易忽略


# go  # golang  # ai  # 配置文件  # bool  #   # 并发  # channel  # 令牌  # 信号量  # 跳过  # 最多  # 是一个  # 多个  # 发信号  # 执行时间  # 也可  # 可在 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?  高端建站三要素:定制模板、企业官网与响应式设计优化  Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)  Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】  Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理  Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】  javascript基本数据类型及类型检测常用方法小结  Laravel如何与Inertia.js和Vue/React构建现代单页应用  大学网站设计制作软件有哪些,如何将网站制作成自己app?  如何快速选择适合个人网站的云服务器配置?  香港服务器WordPress建站指南:SEO优化与高效部署策略  Python图片处理进阶教程_Pillow滤镜与图像增强  Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层  零基础网站服务器架设实战:轻量应用与域名解析配置指南  Laravel如何使用Seeder填充数据_Laravel模型工厂Factory批量生成测试数据【方法】  今日头条微视频如何找选题 今日头条微视频找选题技巧【指南】  免费网站制作appp,免费制作app哪个平台好?  装修招标网站设计制作流程,装修招标流程?  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  如何快速建站并高效导出源代码?  JS中页面与页面之间超链接跳转中文乱码问题的解决办法  个人网站制作流程图片大全,个人网站如何注销?  美食网站链接制作教程视频,哪个教做美食的网站比较专业点?  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  如何在宝塔面板中修改默认建站目录?  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  如何制作一个表白网站视频,关于勇敢表白的小标题?  Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)  jquery插件bootstrapValidator表单验证详解  如何在万网自助建站中设置域名及备案?  JS中对数组元素进行增删改移的方法总结  新三国志曹操传主线渭水交兵攻略  Android实现代码画虚线边框背景效果  网站优化排名时,需要考虑哪些问题呢?  Java类加载基本过程详细介绍  EditPlus中的正则表达式实战(5)  PythonWeb开发入门教程_Flask快速构建Web应用  如何安全更换建站之星模板并保留数据?  Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】  php json中文编码为null的解决办法  php结合redis实现高并发下的抢购、秒杀功能的实例  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  音响网站制作视频教程,隆霸音响官方网站?  如何在七牛云存储上搭建网站并设置自定义域名?  Linux网络带宽限制_tc配置实践解析【教程】  Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置  Laravel如何使用Collections进行数据处理?(实用方法示例)  Android自定义控件实现温度旋转按钮效果  laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法  免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?