开门见山写一个

package main

import (
"fmt"
"github.com/robfig/cron"
"log"
"strings"
"time"
) func CronTask() {
log.Println("******** ******* *******")
} func CronTest() {
log.Println("Starting Cron...") c := cron.New()
c.AddFunc("* * * * * *", CronTask) //2 * * * * *, 2 表示每分钟的第2s执行一次
c.Start() t1 := time.NewTimer(time.Second * ) // ?time.Second * 10 啥意思? *100行吗?
for {
select {
case <-t1.C:
fmt.Println("Time now:", time.Now().Format("2006-01-02 15:04:05")) // 为何要专门制定这个时间
t1.Reset(time.Second * )
}
}
} func main() {
fmt.Println(strings.Repeat("START ", ))
CronTest()
fmt.Println(strings.Repeat("END ", ))
}

核心的定时器代码就3行

c := cron.New()
c.AddFunc("* * * * * *", CronTask)
c.Start()

那后面那些代码时作甚的?

一开始看到示例代码时,有个疑惑,如代码中注释

t1 := time.NewTimer(time.Second * 10)

这里time.Second*10是干啥的? 是否可以写成*100呢, 改了后原来是可以的,那更疑惑了既然都行为啥还要写个这个?

还有后面的for-select-case也是一脸懵逼~~~~

运行代码,从结果反推下原理吧,一次执行结果

START START START START START START START START START START START START START START START
// :: Starting Cron...
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
Time now: -- ::
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
Time now: -- ::
// :: ******** ******* *******

以上是运行的片段,有两大发现

  • 有START START START。。。没有END END END 。。。。:说明了代码在执行时阻塞在定时器里,定时器没有执行完,永远不会执行END
  • Time now打出来的间隔正好是10s

哦,原来time.NewTimer是个定时器,当这个时间间隔完了后再重新打开一个。for-select-case 这一块目的是阻塞流程,不让程序结束。 理解对吗

如果是这样,去掉for-select-case 执行第一个定时器时也可以停10s,是这样吗?试验下:屏蔽掉for-select-case, 输出

START START START START START START START START START START START START START START START
// :: Starting Cron...
END END END END END END END END END END END END END END END

打脸了,看来阻塞主要靠for-select-case实现,那原理是什么呢?

去掉t1.Reset效果咋样呢?

t1 := time.NewTimer(time.Second * ) // ?time.Second * 10 啥意思? *100行吗?
for {
fmt.Println("hihihihi")
select {
case <-t1.C:
fmt.Println("hello")
}
}

输出

START START START START START START START START START START START START START START START
// :: Starting Cron...
hihihihi
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
hello
hihihihi
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******
// :: ******** ******* *******

更蒙了,去掉reset, 运行完第一个定时器10s, 非但没听,还直接执行起来了,没停了

for 循环里的print不是刷刷的一大片,而是和case命中时一期打,看来是时候了解下select-case的原理了

select case

按惯例先上个例子

package main

import (
"fmt"
"strings"
) func SelectTest() {
intChan := make(chan int, )
stringChan := make(chan string, )
intChan <-
stringChan <- "hello" select {
case value := <-intChan:
fmt.Println(value)
case value := <- stringChan:
fmt.Println(value)
}
} func main() {
fmt.Println(strings.Repeat("START ", ))
SelectTest()
fmt.Println(strings.Repeat("END ", ))
}

执行多次可以看到,输出的结果是 123456、"hello"不定

select 语法

每个case都必须是个通信

如果一个通信可进行它就执行,其他被忽略

如果有多个case可执行,就会随机的选择一个执行

如果没有case可执行,如果如果有default,执行default语句;否则就阻塞,直到有某个通信可行

这里还是有很多问题,单开一节弄清楚 select语句

再回到一开始的定时任务

回归正题,定时原理解析

t1 := time.NewTimer(time.Second * )
for {
select {
case <-t1.C:
fmt.Println("Time now:", time.Now().Format("2006-01-02 15:04:05"))
t1.Reset(time.Second * )
}
}
  • 生成一个定时器t1, 执行for循环,一开始定时时间(10s)未到, 也没有阻塞任务,它就阻塞在case中;
  • 定时时间到了,则执行case中语句;
  • 然后又重新恢复定时时长
  • 重新走for循环,还是重复上面的故事

上面代码中尝试把case中的t1.Reset去掉,结果也是定时任务不同的执行,原因是执行case中的语句后,接着执行for循环,由于没有新通信过来(case语句永远无法满足),同时没有default语句,所以同样可以阻塞再次。

相比在case中t1 Reset, t1 Reset更灵活些,因为可以再每次重新满足case时做一些灵活的操作,比如跳出循环,做一些统计打印等。

Golang——Cron 定时任务的更多相关文章

  1. Golang cron 定时任务使用

    1.cron 表达式的基本格式 用过 linux 的应该对 cron 有所了解.linux 中可以通过 crontab -e 来配置定时任务.不过,linux 中的 cron 只能精确到分钟.而我们这 ...

  2. golang的定时任务

    golang的定时任务使用的是cron这个包来解决的 官方文档地址:https://godoc.org/github.com/robfig/cron cron包的基础知识 字段名 是否必须 允许的值 ...

  3. Go cron定时任务的用法

    cron是什么 cron的意思就是:计划任务,说白了就是定时任务.我和系统约个时间,你在几点几分几秒或者每隔几分钟跑一个任务(job),就那么简单. cron表达式 cron表达式是一个好东西,这个东 ...

  4. 使用 cron 定时任务实现 war 自动化发布

    autoRelease.sh #!/bin/sh /home/tomcat/bin/shutdown.sh echo "tomcat stoped" cd /home/tomcat ...

  5. cron定时任务介绍

    什么是cron? Cron是linux系统中用来定期执行或指定程序任务的一种服务或软件.与它相关的有两个工具:crond 和 crontab.crond 就是 cron 在系统内的宿主程序,cront ...

  6. linux ,cron定时任务 备份mysql数据库

    cron 定时任务执行备份脚本文件 backup.sh #!/bin/bash USER="root" PASSWORD="xxxxx" DATABASE=&q ...

  7. 珠峰培训node 珠峰爬虫| cron 定时任务

    1.cron 定时任务 CronJob var CronJob = require('cron').CronJob; // 秒 分钟 时 天

  8. Cron定时任务应用到Thinkphp – 贤生博客

    Cron定时任务应用到Thinkphp 安装crontab: yum install crontabs 关于cron的一些命令: /sbin/service crond start //启动服务 /s ...

  9. linux下的cron定时任务知识梳理

    1 cron定时任务 1.1 cron介绍 为什么需要cron定时任务? 1)cron服务在安装完Linux系统后就默认就存在,主要用来定期执行命令或定期执行指定的应用程序; 2)cron服务默认情况 ...

随机推荐

  1. 关于json语句的相关用法

    json语句: JSON 值可以是: 数字(整数或浮点数)字符串(在双引号中)逻辑值(true 或 false)数组(在中括号中)对象(在大括号中)null 对于json的的对象数组:var site ...

  2. go 内存优化

    一.斐波那切数列优化 package main import ( "time" "fmt" ) const LIM = 41 var fibs [LIM]uin ...

  3. Linux服务器架设篇,Windows中的虚拟机linux上不了外网怎么办?

    1.将电脑的网线口直连路由器内网接口(确保该路由器可以直接正常上网,切记不可以使用宽带连接和无线网连接). 2.在实体机电脑可以上网的前提下,在命令框窗口输入 ipconfig 3.记录下电脑以太网的 ...

  4. Java研发技术学习路线

    Java研发技术成长路线 作为一名Java研发者,深感Java技术的学习是一个漫长过程,从一名Java菜鸟开始,加之持之以恒的耐心和脚踏实地的精神,不间断理论的学习,不停止技术实践,终成为一名技术佼佼 ...

  5. Array(数组)对象-->数组的删除

    1.数组的删除: 用delete操作符删除特定的元素 删除元素的位置只是被留空了,为undefined值 举例:删除下面数组中的第二个元素 var arr = [1,2,3,4,5]; /*删除第二个 ...

  6. scala_spark实践2

    参考:jianshu.com/p/9d2d225c1951 监听socket获取数据,代码如下:这里使用nc -lk 9999 在ip为10.121.33.44的机器上发送消息 object Sock ...

  7. 带你走进神一样的Elasticsearch索引机制

    更多精彩内容请看我的个人博客 前言 相比于大多数人熟悉的MySQL数据库的索引,Elasticsearch的索引机制是完全不同于MySQL的B+Tree结构.索引会被压缩放入内存用于加速搜索过程,这一 ...

  8. 2019应届生,用python爬虫记录自己求职经历,分享求职中的一些坑

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. PS:如有需要Python学习资料的小伙伴可以加点击下方链接自行获取http ...

  9. 8行代码帮你python建立UDP通信

    首先我们要搭建一个UDP通信,首先我们就要知道什么UDP: UDP用户数据报传输协议,它位于TCP/IP协议的传输层,是一种无连接的协议,它发送的报文不能确定是否完整地到达了另外一端.UDP广泛应用于 ...

  10. The equation SGU - 106

    题目链接:https://codeforces.com/problemsets/acmsguru/problem/99999/106 这个题是关于EXGCD特别好的一个题目.题目大意:有一个等式ax+ ...