从0实现分布式任务调度系统--实现cron表达式定时任务

Posted 尚书左仆射

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从0实现分布式任务调度系统--实现cron表达式定时任务相关的知识,希望对你有一定的参考价值。

在Linux下会经常使用cron表达式完成一些定时任务。下面通过git上的一个项目来达到类似的效果。

获取资源:

https://github.com/gorhill/cronexpr

该包一共支持如下7个时间粒度设置,分别是:秒级,分,小时,日期,月份,星期,年份。

各个支持的符号说明如下:

“*” 字符代表所有可能的值。

“-” 表示指定范围。

“,” 表示列出枚举值。

“/” 被用于指定增量。

“L” 表示last,放在“星期”字段中,“FRIL”可以用来表示当月的最后一个周五;放在日期字段中“L”代表该月的最后一天。

“W”只能用在日期字段,是“Weekday”的缩写,用来描叙最接近指定天的工作日(周一到周五)。例如:在day-of-month字段用“15W”指“最接近这个月第15天的工作日”。

“#” 只能在星期字段中使用,后面必须跟一个 1-5 的数字。例如“MON#2”,表示第2个周一。

其他的细节,请参阅该项目的README文档。

 

下面使用Go语言来实现一个简单的定时调用程序:

package main

import(
   "fmt"
   "github.com/gorhill/cronexpr"
   "time"
)

//定义一个任务
type CronJob struct 
   expr *cronexpr.Expression
   nextTime time.Time //通过expr.Next(now)来获得


func main() 
   //需要有一个协程,他定时检查所有的Cron任务,谁过期了就执行谁
   var(
      expr *cronexpr.Expression
      now time.Time
      cronJob *CronJob
      scheduleTable map[string]*CronJob // key:job_name
   )

   scheduleTable = make(map[string]*CronJob)

   //当前时间
   now = time.Now()

   //定义2个cronjob
   expr = cronexpr.MustParse("*/5 * * * * * *")
   cronJob = &CronJob
      expr:expr,
      nextTime:expr.Next(now),
   
   //任务注册到调度表
   scheduleTable["5S_JOB"] = cronJob

   expr = cronexpr.MustParse("*/2 * * * * * *")
   cronJob = &CronJob
      expr:expr,
      nextTime:expr.Next(now),
   
   //任务注册到调度表
   scheduleTable["2S_JOB"] = cronJob

   //启动一个调度协程
   go func() 
      var(
         jobName string
         cronJob *CronJob
         now time.Time
      )
      //定时检查任务调度表
      for 
         now = time.Now()

         for jobName, cronJob = range scheduleTable
            //判断是否过期
            if cronJob.nextTime.Before(now) || cronJob.nextTime.Equal(now)
               //启动一个协程,执行这个任务
               go func(jobName string) 
                  fmt.Println("执行:", jobName)
               (jobName)

               //计算下一次执行时间
               cronJob.nextTime = cronJob.expr.Next(now)
               fmt.Println(jobName, "下次执行时间:", cronJob.nextTime)
            
         
         //睡眠100ms
         select 
         case <- time.NewTimer(100 * time.Millisecond).C://将在100毫秒后返回可读

         
      
   ()

//调用select阻塞主线程,防止进程退出
   select 

   

执行效果如下:

通过类似的方法就可以很方便地实现各个时间粒度的定时任务调度。

以上是关于从0实现分布式任务调度系统--实现cron表达式定时任务的主要内容,如果未能解决你的问题,请参考以下文章

Java的任务调度,使用Cron表达式实现

Golang定时任务简单实现

从0实现分布式任务调度系统--MongoDB基础操作

从0实现分布式任务调度系统--etcd基础操作

几种主流的分布式定时任务,你知道哪些?

Quartz定时任务触发器启动时设置