crontab、watch、systemd timer定时任务

 

1、crontab定时任务

默认最小单位分钟级别,如果要实现秒级通常需要shell sleep等进行辅助控制实现。

例如:每30s执行一次

# 1. 多条crontab 每分钟一次,另一条sleep 30s后执行
# Need these to run on 30-sec boundaries, keep commands in sync.
* * * * *              /path/to/executable param1 param2
* * * * * ( sleep 30 ; /path/to/executable param1 param2 )

#2.单条crontab
* * * * * /bin/bash -l -c "/path/to/executable; sleep 30 ; /path/to/executable"

2、watch定时任务

# 前台执行
# watch
--interval .30 script_to_run_every_30_sec.sh # nohup后台执行 每秒一次输出系统的平均负载到文件
# nohup watch -n 1 'cat /proc/loadavg >> 1.txt' &

3、systemd timer定时任务

systemd timer可以在任何时间粒度上调度程序,理论上可以小到纳秒级别。任务调度上的灵活性远比crontab高,并且不需使用sleep这种蹩脚的方案。systemd定时任务的实现需要两个配置,一个是service,一个是timer。其中service用来定义任务内容,timer用来定义执行周期。下面有个简单的例子,目标是让系统logger每十秒钟输出一次“Hello World”

vim /etc/systemd/system/helloworld.service

[Unit]
Description=Say Hello
[Service] ExecStart
=/usr/bin/logger -i Hello World

vim /etc/systemd/system/helloworld.timer

[Unit]
Description=Say Hello every 10 seconds

[Timer] OnBootSec
=10 OnUnitActiveSec=10 AccuracySec=1ms
[Install] WantedBy
=timers.target

helloworld.timer里并没有声明service的名称,那它和service是如何关联的呢?默认情况下systemd定时器和service的关联是通过相同的名称,如果名称不同则可以在[Timer]配置里通过指定Unit字段来指定。如果想让整个系统使用这个定时器,这两个文件就需要放置在/etc/systemd/system下。如果想给某个用户使用,则放置在~/.config/systemd/user。

如果想让这个定时器立即运行,则可以使用--now参数进行enable,否则,只有等系统重启后,或者用户登录是才会触发运行。

systemctl enable --now helloworld.timer

[Timer]部分里的各个字段的作用如下:

OnBootSec – 系统启动多少秒后开始执行调度

OnUnitActiveSec – 重复调度相关service的时间间隔。就是这行配置决定了跟cron job一样定时调度的动作。

AccuracySec – 定时器精度。 默认是一分钟,跟cron很相似。可以要求的更高,但精度增加会带来更多系统的消耗,更频繁的唤醒CPU。上面的配置里写的是1ms,显然不是个聪明的决定。通常我们可以把它设置为 1(1秒),对于我们这样低于1分钟时间粒度的定时器的精度要求已经够用了。也是因为如此,我们会看到,实际程序运行时输出“Hello World”消息的时间经常会延迟1秒左右。如果你认为这一秒左右的延迟不是问题,那就应该这样设定。

systemd timer可以分为相对时间周期和日历时间周期。相对执行周期的参考时间为第一次执行的时间,之后按照周期执行。日志时间周期执行则需要通过OnCalendar来设定。

例如:

OnCalendar=*-*-* *:*:00,10,20,30,40,50

 

posted @ 2020-08-23 01:57  JeromePowell  阅读(433)  评论(0编辑  收藏  举报