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

从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 {
 
   }
}
执行效果如下:

通过类似的方法就可以很方便地实现各个时间粒度的定时任务调度。
————————————————
版权声明:本文为CSDN博主「尚书左仆射」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/wzxq123/article/details/103936344

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值