我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Golang如何实现CronJob

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Golang如何实现CronJob

这篇文章主要讲解了“Golang如何实现CronJob”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Golang如何实现CronJob”吧!

引言

最近做了一个需求,是定时任务相关的。以前定时任务都是通过 linux crontab 去实现的,现在服务上云(k8s)了,尝试了 k8s 的 CronJob,由于公司提供的是界面化工具,使用、查看起来很不方便。于是有了本文,通过一个单 pod 去实现一个常驻服务,去跑定时任务。

经过筛选,选用了cron这个库,它支持 linux cronjob 语法取配置定时任务,还支持@every 10s、@hourly 等描述符去配置定时任务,完全满足我们要求,比如下面的例子:

package mainimport ("fmt""github.com/natefinch/lumberjack""github.com/robfig/cron/v3""github.com/sirupsen/logrus")type CronLogger struct {clog *logrus.Logger}func (l *CronLogger) Info(msg string, keysAndValues ...interface{}) {l.clog.WithFields(logrus.Fields{"data": keysAndValues,}).Info(msg)}func (l *CronLogger) Error(err error, msg string, keysAndValues ...interface{}) {l.clog.WithFields(logrus.Fields{"msg":  msg,"data": keysAndValues,}).Warn(err.Error())}func main() {logger := logrus.New()_logger := &lumberjack.Logger{Filename:   "./test.log",MaxSize:    50,MaxAge:     15,MaxBackups: 5,}logger.SetOutput(_logger)logger.SetFormatter(&logrus.JSONFormatter{DisableHTMLEscape: true,})c := cron.New(cron.WithLogger(&CronLogger{clog: logger,}))c.AddFunc("*/5 * * * *", func() {fmt.Println("你的流量包即将过期了")})c.AddFunc("*/2 * * * *", func() {fmt.Println("你的转码包即将过期了")})c.Start()for {select {}}}

使用了 cronjob、并结合了 golang 的 log 组建,输出日志到文件,使用很方便。

但是,在使用过程中,发现还有些不足,缺少某些功能,比如我很想使用的查看任务列表。

类库介绍

扩展性强

此类库扩展性挺强,通过 JobWrapper 去包装一个任务,NewChain(w1, w2, w3).Then(job),相关实现如下:

type JobWrapper func(Job) Jobtype Chain struct {    wrappers []JobWrapper}func NewChain(c ...JobWrapper) Chain {    return Chain{c}}func (c Chain) Then(j Job) Job {    for i := range c.wrappers {        j = c.wrappers[len(c.wrappers)-i-1](j)    }    return j}

比如当前脚本如果还没有执行完,下次任务时间又到了,就可以通过如下默认提供的 wrapper 去避免继续执行。可以看到最后执行的任务 j.Run() 被包装在了一个函数闭包中,并且根据闭包中的 channel 去判断是否执行,避免重复执行。首次执行的时候,容量为 1 的 channel 中已经有数据了,重复执行时,channel 无数据,默认跳过,等上次任务执行完成后,又像 channel 中写入一条数据,下次 channel 可以读出数据,又可以执行任务了:

func SkipIfStillRunning(j Job) Job {    var ch = make(chan struct{}, 1)    ch <- struct{}{}    return FuncJob(func() {        select {        case v := <-ch:            defer func() { ch <- v }()            j.Run()        default:            // "skip"        }    })}

主流程

cron 主流程是启动一个协程,里面有双重 for 循环,下面我们来一起分析一下。

定时器

第一层循环,首先计算下次最早执行任务的时间跟当前时间间隔 gap,然后设置定时器为 gap,这里很巧妙,定时器间隔不是 1s/次,而是跟下次任务的时间相关,这样就避免了无用的定时器循环,也让执行时间更精准,不存在设置小了浪费资源,设置大了误差大的情况。接下来进入第二层循环。

sort.Sort(byTime(c.entries))timer = time.NewTimer(c.entries[0].Next.Sub(now))

事件循环

事件循环中,包含了很多事件,比如 添加任务、停止、移除任务,当 cron 启动之后,这些任务都是异步的。比如添加任务,不会直接将任务信息写入内存中,而是进入事件循环,加入之后,重新计算第一二层循环,避免了正在修改任务信息,又执行任务信息,然后出错的情况。

有人可能会问了,为何不在事件中加锁,这样也能避免内存竞争。我想说,我们执行的是脚本任务,有的事件可能很长,可能会阻塞有些事件,所以这些事件都放在循环中,避免了加锁,也满足了要求。

for {    select {    case now = <-timer.C:        // 执行任务    case newEntry := <-c.add:        // 添加任务    case replyChan := <-c.snapshot:        // 获取任务信息    case <-c.stop:        //  停止任务    case id := <-c.remove:        // 移除任务    }    break}

类库改造

在了解了项目的基本情况之后,对项目做了部分改造,方便使用。

打印任务列表信息

在主循环汇总加入了信号量监听,当触发信号量 SIGUSR1,将任务信息输出到日志:

usrSig := make(chan os.Signal, 1)signal.Notify(usrSig, syscall.SIGUSR1)for {select {case <-usrSig:// 启动单独的协程去打印定时任务执行信息continue}break}

根据名称移除脚本

目前脚本只能根据脚本 id 去移除要执行的任务,执行过程中,也不能通过命令去移除任务,不是太方便。比如有个脚本马上要执行了,但是该脚本发现问题了,这时候生产环境的话,就需要更新代码,然后重启服务去下线脚本任务,这时候,黄花菜可能都凉了。

所以我也是通过信号量,来处理运行之后,运行中移除任务的问题,收到信号量之后,读取文件中的内容,根据命令去处理 runing 中的内存:

usrSig2 := make(chan os.Signal, 1)signal.Notify(usrSig2, syscall.SIGUSR2)......case <-usrSig2:actionByte, err := os.ReadFile("/tmp/cron.action")...... //校验命令正确性action := strings.Fields(string(actionByte))switch action[0] {case "removeTag":timer.Stop()now = c.now()c.removeEntryByTag(action[1])c.logger.Info("removedByTag", "tag", action[1])}......

改造效果

由于原项目已经 2 年多没有个更新过了,就算发起 pr 估计也不会被处理,所以 fork 一份放在了这里aizuyan/cron进行改造,下面是改进之后的代码:

package mainimport (// 加载配置文件"fmt""github.com/aizuyan/cron/v3")func main() {c := cron.New(cron.WithLogger(cron.DefaultLogger))c.AddFuncWithTag("流量包过期", "*/5 * * * *", func() {fmt.Println("你的流量包即将过期了")})c.AddFuncWithTag("转码包过期", "*/2 * * * *", func() {fmt.Println("你的转码包即将过期了")})c.Start()for {select {}}}

对每个定时任务增加了一个名称标识,当任务启动后,当我们执行 kill -SIGUSR1 <pid> 的时候,会看到 stdout 输出了运行的任务列表信息:

+----+------------+-------------+---------------------+---------------------+
| ID |    TAG     |    SPEC     |        PREV         |        NEXT         |
+----+------------+-------------+---------------------+---------------------+
|  2 | 转码包过期 | */2 * * * * | 0001-01-01 00:00:00 | 2023-04-02 17:22:00 |
|  1 | 流量包过期 | */5 * * * * | 0001-01-01 00:00:00 | 2023-04-02 17:25:00 |
+----+------------+-------------+---------------------+---------------------+

执行 kill -SIGUSR2 <pid>,移除转码包过期任务,避免了使用 ID 容易出错的问题。

cat /tmp/cron.action removeTag 转码包过期// {"data":["tag","转码包过期"],"level":"info","msg":"removedByTag","time":"2023-04-02T18:32:56+08:00"}

感谢各位的阅读,以上就是“Golang如何实现CronJob”的内容了,经过本文的学习后,相信大家对Golang如何实现CronJob这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Golang如何实现CronJob

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Golang如何实现CronJob

这篇文章主要讲解了“Golang如何实现CronJob”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Golang如何实现CronJob”吧!引言最近做了一个需求,是定时任务相关的。以前定时任
2023-07-05

Golang实现CronJob(定时任务)的方法详解

这篇文章主要为大家详细介绍了Golang如何通过一个单pod去实现一个常驻服务,去跑定时任务(CronJob),文中的示例代码讲解详细,需要的可以参考下
2023-05-14

k8s中job与cronjob如何使用

本篇内容介绍了“k8s中job与cronjob如何使用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、前言job,顾名思义就是任务,job
2023-07-05

golang如何实现hash

本篇内容介绍了“golang如何实现hash”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!哈希(Hash)指的是将任意长度的二进制串映射为固
2023-07-06

golang map如何实现

本文小编为大家详细介绍“golang map如何实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“golang map如何实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。哈希表的概念哈希表是一种以键值对存储数
2023-07-05

golang如何实现map

本文小编为大家详细介绍“golang如何实现map”,内容详细,步骤清晰,细节处理妥当,希望这篇“golang如何实现map”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。随着大数据时代的到来和云计算技术的普及,数
2023-07-05

golang接口如何实现

今天小编给大家分享一下golang接口如何实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。在golang中,可以使用“ty
2023-07-04

golang缓存如何实现

这篇文章主要讲解了“golang缓存如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“golang缓存如何实现”吧!一、什么是缓存?缓存是一种提高数据读写性能的技术,类比于日常生活中的翻
2023-07-06

golang异步如何实现

本篇内容主要讲解“golang异步如何实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“golang异步如何实现”吧!在golang中,异步是指不按照代码顺序执行,一个异步过程的执行将不再与原有
2023-07-04

golang如何实现多态

go 中没有传统多态,但可以利用接口和反射实现类似效果:定义接口,明确方法集。创建多个类型,实现该接口。使用反射,动态调用方法,无需了解具体类型。Go 中实现多态如何实现?Go 中没有传统意义上的多态,但可以使用接口和反射机制来实现类似
golang如何实现多态
2024-04-21

golang如何实现并发

要实现并发,可以通过关键字”go“来启动一个新的”goroutine“:1、定义一个函数”doSomething“,编写具体的并发任务逻辑;2、定义”main“函数,通过”go“关键字启动新的”goroutine“,而主程序继续执行其他逻辑
golang如何实现并发
2023-12-12

golang如何实现队列

golang可以使用标准库中的”container/list“包实现队列:1、使用import关键字导入包;2、定义函数”main“;3、通过”list.New()“函数创建了一个新的队列q;4、使用”PushBack()“方法向队列中添加
golang如何实现队列
2023-12-12

golang如何实现链表

实现链表的方法:1、定义了一个Node结构体来表示链表的节点,每个节点包含一个数据项和一个指向下一个节点的指钁;2、定义了一个LinkedList结构体来表示链表本身,其中包含一个指向链表头节点的指针;3、实现了两个方法,append用于在
golang如何实现链表
2023-12-14

golang如何实现高并发

Golang通过Goroutine和Channel来实现高并发。Goroutine是Golang中轻量级的线程,可以同时执行多个Goroutine,且切换开销很小。通过关键字go可以创建一个新的Goroutine。例如:go func(
2023-10-23

Golang如何实现单链表

今天小编给大家分享一下Golang如何实现单链表的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1. 定义节点// Node
2023-07-05

golang如何实现文件锁

本篇内容主要讲解“golang如何实现文件锁”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“golang如何实现文件锁”吧!在golang中,可以利用sync包的api来实现文件锁。文件锁(flo
2023-07-04

Golang中如何实现枚举

这篇文章主要介绍了Golang中如何实现枚举的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Golang中如何实现枚举文章都会有所收获,下面我们一起来看看吧。在编程领域里,枚举用来表示只包含有限数量的固定值的类型
2023-06-29

如何使用Golang实现NFT

今天小编给大家分享一下如何使用Golang实现NFT的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。一、什么是NFTNFT是一
2023-07-06

golang熔断器如何实现

这篇“golang熔断器如何实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“golang熔断器如何实现”文章吧。熔断器像是
2023-06-26

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录