如何实现一个简单高效的定时任务系统(With Golang)

场景

有一个需求是用户可以自己设定时间, 定时推送给用户通知, 例如邮件订阅每日新闻.

这个需求应该并不刁钻, 按理说在网上有现成的框架或者工具. 由于我正在使用的语言是Golang, 所以我希望这个框架也是Golang实现的, 故搜索关键词是 Golang 定时任务 / Golang cron:

有的是一个系统:

有的是一个库:

其中不乏Star上千的优秀项目, 却发现并不适合我的业务需求:

  • 上百万量级的定时任务支持. 而上面的项目没有说性能 也没有指出如何扩展性能.
  • 任务入库, 可以增删改查. 这点除了robfig/cron的框架, 上面所述的系统都是支持的, 但他们要么不支持API, 要么还没写文档.....
  • 简单. 至少在我看来以上的系统都没有提供充足的文档让它看起来简单清晰, 特别是故障恢复和如何提升性能部分.

那么是时候自个儿思考下如何做一个适合项目的定时任务系统了. 正好, 这也是我去了解分布式与高可用的入门题.

目的

分布式

这个程序应当很好扩展出分布式架构, 最好是去中心化的, 因为我能力有限, 不想引入像etcd这样繁琐的依赖.

简单

不重复造轮子, 尽量使用开源项目. 一是更快完成需求, 二是这样方便维护

重新思考

研究上述优秀项目后启发也很大, 抱着少重复造轮子的想法, 所以对于已有的东西能用则用. 故

  • cron库能够被很好的使用, 用于计算下一次运行时间.
  • 用消息队列来保证任务执行的可靠性与性能

一个任务执行一次的流程如下:

  1. 计算任务的下一次执行时间并存放在DB
  2. 调度器每1s取出需要被执行的任务(run_at < now)发送给消息队列, 并更新下一次执行时间
  3. 消息队列将任务消息分发到执行任务的机器
  4. 消费者执行任务.

由于我们引入了消息队列, 所以我们不再需要设计整个系统中最复杂的部分: 任务消息分发. 这交给值得信赖的NSQ来做十分合适, 当然如果你想使用其他消息队列如kafka也可以.

接下来是详细设计

数据库我们可以设计如下

id cron next_run_at topic body
1 * * * 10 10 1111111 send_email {"to":"zbysir@qq.com","content":"hello world"}

其中

  • cron: cron表达式, 更多请看
  • next_run_at: 使用这个cron库能方便的计算出下次运行时间.
  • topic: 也就是Nsq消息队列中的topic, 可以用于当做要执行的任务类型, 如这里是用来发送邮件.
  • body: 任何消息体.

我们可以给next_run_at添加索引已获得更好的性能, 因为我们需要使用这个字段排序.

现在只需要写一个简单的程序, 每一秒去读取一下需要执行(next_run_at < now)的任务, 并Publish到NSQ就好了.

关于NSQ如何使用与部署, 如果你不了解的话, 可以先去官网看一看, 或者是我的另一篇文章: Nsq从入门到实践, 欢迎吐槽.

就是这样简单, 你不需要找一个专门为定时任务而设计的框架, 使用这个方案 就算加上增删改查功能 代码也不超过几百行.

如何升级性能

不用太担心性能, 整个项目的性能瓶颈只取决于MySql的读取速度, 而对于MySql的优化建议网上一搜一大把, 比如分库: 我们可以将发送邮件 和 发送短信这两个任务分为两个库, 或者将用户id<10000的定时任务分一个表, id>10000分一个表, 这和写你的业务代码一样.

而NSQ的性能我们也无需担心, 反正都是可以无限水平扩展的.

我建议的架构拓扑图如下:

image.png

如果你了解NSQ的话, 你会发现这其实就是NSQ建议的分布式部署方式. 整个系统非常简单, 没有任何需要额外理解的逻辑.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容

  • 本文为转载,原文:Golang 简单爬虫实现 · 定时任务 介绍 通过前一篇文章,我们已经实现了简单的爬虫,爬取小...
    ChainZhang阅读 1,598评论 0 5
  • golang 实现定时服务很简单,只需要简单几步代码便可以完成,不需要配置繁琐的服务器,直接在代码中实现。 使用 ...
    正在修炼的西瓜君阅读 12,471评论 0 8
  • 有时候 爱情好像离自己越来越遥远 当生命的终点来临 当我和你的路出现分叉路 也许是到了该放弃的下一秒 我想去怀念 ...
    往朔阅读 257评论 0 2
  • 人死之后 还能否感受到外界的压力? 不能的话 土包和金字塔又有何异? 若能 诺大的重压岂不让灵魂窒息? 那该不是用...
    楸木行枷阅读 175评论 0 1
  • 一个人,在飞 有落日,有余晖 苍凉着脚下的 层叠树木,满眼翠微 你在笑我孤独么 也许吧,是这样 是我一人在飞 可你...
    俞壹阅读 286评论 2 8