【Kubernetes】CronJob 执行时间存在大量延迟

在这里插入图片描述
最近在迁移集群CronJob的时候,发现了一个问题:CronJob执行大概差了八个小时。当前的(指写文章)时间是5.18 19:36分,往前推45个小时,大概是5.16 22:36分,和我预设CronJob表达式子里写明的
0 14 16 * * 16号14:00相差了八个小时。

今天来研究一下这个小时究竟问题出在哪。

从八个小时上来判断,正好是UTC +0 和 UTC +8的时差,也正好是我们的美国服务器和上海时间的时差。所以猜测上是集群设置时区的问题。

查了一下文档,发现CronJob的执行时间是根据kube-controller-manager的时间来控制
在这里插入图片描述
所以看了一下这个pod上的时区设置是UTC +0
在这里插入图片描述
然后看了一下 在CronJob表达式的业务逻辑,指定了上海这个时区。

        Date date = new Date(time);
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        simpleDateFormat.setTimeZone(TimeZone.getTimeZone("Asia/Shanghai"));

这个问题的链路就是 微服务A制定了上海市UTC+8作为Cron表达式,然后将这个Cron表达式发送给kube-api由kube-crontroller-manager来执行控制。而kube-crontroller-manager的时区认定在了UTC+0。

举个例子,处于UTC+8的微服务A说你在 16号 8点钟来执行这个任务,但是A并没有考虑美国服务区的时区。Kube-Controller-manager认为这个8点钟是对于自己UTC+0来说的时区,所以让微服务A多等待了八个小时。

最后解决办法倒是很简单,直接把上面的业务代码换成对应Kube-Controller-manager对应的时区。

        Date date = new Date(time);
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        simpleDateFormat.setTimeZone(TimeZone.getTimeZone(ZoneOffset.UTC));

但这只是一时之计,撤掉解决的思路大概现在有两种

  1. 站在SpringBoot 约定大于配置的角度上想,如果几方决定使用UTC+0的时区的话,应该和所有达成一致。否则不应该随便更换Kube-Controller-manager的时区。
  2. 站在Client-Server上来看,我一个调用方为啥要考虑业务信息之外的逻辑,应该直接将时间戳发送给被调用方,让被调用方通过本地服务器配置来适配时区。
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Golang 中,你可以使用第三方库,如 `robfig/cron` 或 `go-cron/cron` 来实现 cron 作业的延迟执行。 使用 `robfig/cron` 库的示例代码如下: ```go package main import ( "fmt" "time" "github.com/robfig/cron" ) func main() { c := cron.New() // 定义一个 cron 作业 job := cron.NewChain( cron.DelayIfStillRunning(cron.DefaultLogger), ).Then(cron.FuncJob(func() { fmt.Println("Hello, World!") })) // 设置作业的调度规则 schedule := "*/1 * * * * *" // 每秒执行一次 c.AddJob(schedule, job) // 启动 cron 作业 c.Start() // 延迟程序退出 time.Sleep(5 * time.Second) // 停止 cron 作业 c.Stop() } ``` 在上面的示例中,我们使用了 `cron.DelayIfStillRunning` 方法来延迟执行作业,这样即使上一个作业还在执行中,下一个作业也会按计划延迟执行。 另外,如果你使用的是 `go-cron/cron` 库,你可以使用 `cron.Delay` 方法来实现延迟执行。具体代码示例如下: ```go package main import ( "fmt" "time" "github.com/go-cron/cron" ) func main() { c := cron.New() // 定义一个 cron 作业 job := cron.NewChain( cron.Delay(5*time.Second), ).Then(cron.FuncJob(func() { fmt.Println("Hello, World!") })) // 设置作业的调度规则 schedule := "*/1 * * * * *" // 每秒执行一次 c.AddJob(schedule, job) // 启动 cron 作业 c.Start() // 延迟程序退出 time.Sleep(10 * time.Second) // 停止 cron 作业 c.Stop() } ``` 以上是两个常用的库来实现 golang cron 延迟执行的方法,你可以根据自己的实际需求选择使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值