Задание на время в Голанге: выход на яму и окончательное решение

Go

предисловие

Национальный день не имеет ничего общего, поэтому я реконструировал центр планирования запланированных задач, который раньше был с открытым исходным кодом.

Я столкнулся с некоторыми ямами crontab в течение периода

CronTab

Давайте посмотрим на формат времени crontab,

SpecСсылаться наbeego toolboxпод модульcrontabФормат композиции:

//前6个字段分别表示:
//       秒钟:0-59
//       分钟:0-59
//       小时:1-23
//       日期:1-31
//       月份:1-12
//       星期:0-6(0 表示周日)

//还可以用一些特殊符号:
//       *: 表示任何时刻
//       ,: 表示分割,如第三段里:2,4,表示 2 点和 4 点执行
//       -:表示一个段,如第三端里: 1-5,就表示 1 到 5 点
//       /n : 表示每个n的单位执行一次,如第三段里,*/1, 就表示每隔 1 个小时执行一次命令。也可以写成1-23/1.
/////////////////////////////////////////////////////////
//  0/30 * * * * *                        每 30 秒 执行
//  0 43 21 * * *                         21:43 执行
//  0 15 05 * * *                         05:15 执行
//  0 0 17 * * *                          17:00 执行
//  0 0 17 * * 1                          每周一的 17:00 执行
//  0 0,10 17 * * 0,2,3                   每周日,周二,周三的 17:00和 17:10 执行
//  0 0-10 17 1 * *                       毎月1日从 17:00 到 7:10 毎隔 1 分钟 执行
//  0 0 0 1,15 * 1                        毎月1日和 15 日和 一日的 0:00 执行
//  0 42 4 1 * *                          毎月1日的 4:42 分 执行
//  0 0 21 * * 1-6                        周一到周六 21:00 执行
//  0 0,10,20,30,40,50 * * * *            每隔 10 分 执行
//  0 */10 * * * *                        每隔 10 分 执行
//  0 * 1 * * *                           从 1:0 到 1:59 每隔 1 分钟 执行
//  0 0 1 * * *                           1:00 执行
//  0 0 */1 * * *                         毎时 0 分 每隔 1 小时 执行
//  0 0 * * * *                           毎时 0 分 每隔 1 小时 执行
//  0 2 8-20/3 * * *                      8:02,11:02,14:02,17:02,20:02 执行
//  0 30 5 1,15 * *                       1 日 和 15 日的 5:30 执行

PS:beegoМодуль синхронизации более мощный и поддерживаетвторой уровеньзадача на время

Теперь предположим, что текущее время равно21:11:05, как определитькаждые 5 минутВременная задача, которая выполняется в цикле?

Большинство поисковых запросов в Интернете выглядят так:

0 */5 * * * *

Хорошо, давайте запустим запланированную задачу и посмотрим, что произойдет:

2019/10/05 21:11:05:087825 local [INFO] ------ 定时任务: [Test] 加载成功 ------ cron.go:155
2019/10/05 21:12:00:007810 local [DEBUG] 定时任务: [Test] 成功获取Redis分布式锁, 开始执行调度任务 delivery.go:90
2019/10/05 21:12:00:036411 local [DEBUG] 定时任务: [Test] 调度成功 ... ...
2019/10/05 21:17:00:004286 local [DEBUG] 定时任务: [Test] 成功获取Redis分布式锁, 开始执行调度任务 delivery.go:90
2019/10/05 21:17:00:010506 local [DEBUG] 定时任务: [Test] 调度成功 ... ...


Странно, почему миссия не21:16:05беги, но21:12:00он начинает работать

Онлайн примерно интерпретируется как:

Настройку N следует полностью убрать на 60: Если N можно разделить на 60, то она будет выполняться раз в N минут, а 60/n выполняется один раз, если N нельзя разделить на 60, то Можно реализовать пока убрав и всю точку (с остатком 0).

Как решить?

  • второй

    Секунда (первая цифра): заменить на текущую секунду/60, которая будет выполняться каждые 60 секунд после текущей секунды;

  • Минута(Заменять только в следующем формате):

    */?

    Минуты (второе место): заменить на текущую минуту/каждую минуту и ​​выполнять каждую минуту после текущей минуты;

  • Час(Заменять только в следующем формате):

    */?

    Час (третья цифра): заменить на текущий час/каждый час и каждый час после текущего часа;

Пример:каждые 7 минут,каждые 9 минут,каждые 14 минут,каждые 27 минут

от21:16:05Например

0 */7 * * * *  替换为: 5/60 16/7 * * * *
0 */9 * * * *  替换为: 5/60 16/9 * * * *
0 */14 * * * *  替换为: 5/60 16/14 * * * *
0 */27 * * * *  替换为: 5/60 16/27 * * * *
Кодовое решение:
func ConvertSecond(spec string) string {
	if spec != "" {
		_spec := strings.Split(spec, " ")
		hour, minute, second := time.Now().Clock()

		// second处理
		_second := strconv.FormatInt(int64(second), 10) + "/60"
		spec = strings.Replace(spec, _spec[0], _second, 1)

		// minute处理(类似: */5)
		if strings.HasPrefix(_spec[1], "*/") {
			_minute := strconv.FormatInt(int64(minute), 10) + "/" + strings.Trim(_spec[1], "*/")
			spec = strings.Replace(spec, _spec[1], _minute, 1)
		}

		// hour处理(类似: */5)
		if strings.HasPrefix(_spec[2], "*/") {
			_hour := strconv.FormatInt(int64(hour), 10) + "/" + strings.Trim(_spec[2], "*/")
			spec = strings.Replace(spec, _spec[2], _hour, 1)
		}

		return spec
	}

	return ""
}

хорошо, давайте снова запустим запланированную задачу, чтобы увидеть результаты:

2019/10/05 21:18:38:333458  local [INFO] ------ 定时任务: [Test] 加载成功 ------ cron.go:155
2019/10/05 21:23:38:002390  local [DEBUG] 定时任务: [Test] 成功获取Redis分布式锁, 开始执行调度任务 delivery.go:90
2019/10/05 21:23:38:009905 local [DEBUG] 定时任务: [Test] 调度成功 ... ...
2019/10/05 21:28:38:008273 local [DEBUG] 定时任务: [Test] 成功获取Redis分布式锁, 开始执行调度任务 delivery.go:90
2019/10/05 21:28:38:011781 local [DEBUG] 定时任务: [Test] 调度成功 ... ...

В соответствии с ожиданиями