1. Linux任务计划、周期性任务执行

1. Linux任务计划、周期性任务执行

1.1. 未来的某时间点执行一次任务

at 指定时间点,执行一次性任务
batch 系统自行选择空闲时间去执行此处指定的任务

1.2. 周期性运行某任务

cron

2. at任务

rpm包:at

at 命令:at [option] TIME

常用选项:

-V 显示版本信息 -t time 时间格式 [[CC]YY]MMDDhhmm[.ss] -l 列出指定队列中等待运行的作业;相当于atq -d 删除指定的作业;相当于atrm -c #(编号) :查看具体作业任务 -f /path/file 指定的文件中读取任务 -m 当任务被完成之后,将给用户发送邮件,即使没有标准输出
  • 注意:作业执行命令的结果中的标准输出和错误以邮件通知给相关用户

2.1. TIME格式:定义出什么时候进行 at 这项任务的时间

HH:MM [YYYY-mm-dd] noon, midnight, teatime(4pm) tomorrow now+#{minutes,hours,days, OR weeks}
  1. HH:MM 02:00 在今日的 HH:MM 进行,若该时刻已过,则明天此时执行任务
  2. HH:MM YYYY-MM-DD 02:00 2016-09-20 规定在某年某月的某一天的特殊时刻进行该项任务
  3. HH:MM[am|pm] [Month] [Date]
    • 04pm March 17
    • 17:20 tomorrow
  4. HH:MM[am|pm] + number [minutes|hours|days|weeks] 在某个时间点再加几个时间后才进行该项任务
    • now + 5 min
    • 02pm + 3 days
    • 注意:测试中发现如果设置的时间小于当前时间点(比如现在时间3点,执行命令2点)则它会明天执行(即使此时把时间改为1点59分它也不会马上执行了,因为还有日期的关系,别忘记了)需要注意这一点(crontab中也是这个原理)。

2.2. 执行方式:

  1. 交互式
  2. 输入重定向(多行)
  3. 管道 |
  4. at –f file
  • 它依赖于atd服务,需要启动才能实现at任务

    rpm -ql at systemctl status|start atd.service at DATE<<EOF :一行一个任务 xxx xxx EOF at -l
  • 注意at交互式的话需要用ctrl+d结束。

at队列存放在/var/spool/at目录中 /etc/at.{allow,deny}控制用户是否能执行at任务

  • 白名单:/etc/at.allow 默认不存在,只有该文件中的用户才能执行at命令
  • 黑名单:/etc/at.deny 默认存在,拒绝该文件中用户执行at命令,而没有在此黑名单文件中的使用者则可执行

2.2.1. 注意点:

  1. 如果存在了白名单,则黑名单无效了,如果两个文件都不存在,只有 root 可以执行 at 命令。
  2. 直接用at命令写的计划任务,在系统关机重启后丢失,不能保存。
  3. at 中的计划任务命令中的标准输出不会在屏幕上打印,它会自动写入邮件中发送。用-m可以让没有标准输出的命令也发邮件,为了证明此at任务确实执行了

3. 周期性任务计划cron

  • 相关的程序包:
- `cronie`:主程序包,提供crond守护进程及相关辅助工具 - `cronie-anacron`:cronie的补充程序,用于监控cronie任务执行状况,如cronie中的任务在过去该运行的时间点未能正常运行,则anacron会随后启动一次此任务 - `crontabs`:包含CentOS提供系统维护任务
  1. 确保crond守护处于运行状态:
    CentOS 7:
     systemctl status crond
    CentOS 6:
     service crond status
  2. 计划周期性执行的任务提交给crond,到指定时间会自动运行系统cron任务:系统维护作业
    /etc/crontab :尽量不要改这个系统级定义的计划任务文件,而是自己定义,用命令crontab -e (参照格式写,其中user-name无需再写)
  3. 用户cron任务:
    crontab 命令
    日志:/var/log/cron ,其中记录了计划任务是否执行
    • 注意rpm -ql cronie 或者 rpm -qf /usr/bin/crontab,查看信息。然后ll /usr/bin/crontab 可以看到它有suid权限,因此用户可以定义计划任务。

3.1. 系统级cron任务:/etc/crontab

注释行以 # 开头

详情参见 man 5 crontab

\# Example of job definition: \# .---------------- minute (0 - 59) \# | .------------- hour (0 - 23) \# | | .---------- day of month (1 - 31) \# | | | .------- month (1 - 12) OR jan,feb,mar,apr ... \# | | | | .---- day of week (0 - 6) (Sunday=0 or 7) OR sun,mon,tue,wed,thu,fri,sat \# | | | | | \# * * * * * user-name command to be executed

例如:晚上9点10分运行echo命令

10 21 * * * wang /bin/echo "Howdy!"

3.2. 时间表示法:

(1) 特定值 给定时间点有效取值范围内的值 (2) * 给定时间点上有效取值范围内的所有值 表示“每...” (3) 离散取值 #,#,# (4) 连续取值 #-# (5) 在指定时间范围上,定义步长 /#: #即为步长

3.3. 对应时间格式

@形式时间形式
@rebootRun once after reboot
@yearly0 0 1 1 *
@annually0 0 1 1 *
@monthly0 0 1 * *
@weekly0 0 * * 0
@daily0 0 * * *
@hourly0 * * * *

示例:每3小时echo和wall命令

0 */3 * * * centos /bin/echo “howdy”; wall “welcome to chansha!”

3.4. 系统的计划任务:

/etc/crontab 配置文件 /etc/cron.d/ 文件夹内的配置文件 /etc/cron.hourly/ 脚本 /etc/cron.daily/ 脚本 /etc/cron.weekly/ 脚本 /etc/cron.monthly/ 脚本 /var/log/cron 执行日志 /var/spool/cron/USERNAME :crontab -e命令修改后的用户文件真正存放位置 mail 执行的输出信息会以邮件方式发送

4. anacron系统

 运行计算机关机时cron不运行的任务,CentOS6以后版本取消anacron服务,由crond服务管理

 对笔记本电脑、台式机、工作站、偶尔要关机的服务器及其它不一直开机的系统很重要对很有用

配置文件:/etc/anacrontab,负责执行/etc/cron.daily /etc/cron.weekly /etc/cron.monthly中系统任务

  • 字段1:如果在这些日子里没有运行这些任务……
  • 字段2:在重新引导后等待这么多分钟后运行它
  • 字段3:任务识别器,在日志文件中标识
  • 字段4:要执行的任务

/etc/cron.hourly/0anacron执行

当执行任务时,更新/var/spool/anacron/cron.daily文件的时间戳和其内容,它的内容就是CCYYMMDD

5. 管理临时文件

CentOS6使用/etc/cron.daily/tmpwatch定时清除临时文件

CentOS7使用systemd-tmpfiles-setup服务实现

5.1. systemd-tmpfiles-setup服务

命令:

  • /usr/bin/systemd-tmpfiles [OPTIONS...] [CONFIGFILE...]
  • 例:systemd-tmpfiles –clean|remove|create configfile

主要清除 /tmp (以及/usr/tmp) 和 /var/tmp 中的文件

查看帮助:

  • man systemd-tmpfiles
  • man tmpfiles.d

配置文件:

/etc/tmpfiles.d/*.conf /run/tmpfiles.d/*.conf /usr/lib/tmpfiles/*.conf

比如打开一个配置文件:

# cat /usr/lib/tmpfiles.d/tmp.conf d /tmp 1777 root root 10d d /var/tmp 1777 root root 30d

6. 用户计划任务:/var/spool/cron/USERNAME

crontab命令定义

  • 每个用户都有专用的cron任务文件:/var/spool/cron/USERNAME
crontab命令: crontab [-u user] [-l | -r | -e] [-i] -l 列出所有任务 -e 编辑任务 :谁执行就是以谁为用户来执行,因此user-name这项不用再写。同时在执行它的时候会有一个临时文件,执行完之后便会消除掉它。 -r 移除所有任务 -i 同-r一同使用,以交互式模式移除指定任务 -u user 仅root可运行,指定用户管理cron任务

控制用户执行计划任务:

/etc/cron.{allow,deny}

7. 总结at和crontab

  1. 一次性作业使用 at
  2. 重复性作业使用crontab
操作at命令crontab命令
Createat timecrontab -e
Listat -lcrontab -l
Detailsat -c jobnumcrontab -l
Removeat -d jobnumcrontab -r
EditN/Acrontab -e
  • 没有被重定向的输出会被邮寄给用户
  • root能够修改其它用户的作业

注意:运行结果的标准输出和错误以邮件通知给相关用户

  • (1) COMMAND > /dev/null
  • (2) COMMAND &> /dev/null

7.1. 注意点:

  1. 对于cron任务来讲,%有特殊用途;

    • 如果在命令中要使用%(比如date),则需要转义(用\%的方式)
    • 或者将%放置于单引号中,则可不用转义.
    • 或者把需要%的命令它写入脚本中,然后执行脚本即可
  2. atcrontab在 测试中发现如果设置的时间小于当前时间点(比如现在时间3点,执行命令2点)则它会明天执行(即使此时把时间改为1点59分它也不会马上执行了,因为还有日期的关系,别忘记了)需要注意这一点

  3. 写入计划任务的命令的标准输出尽量都扔到垃圾箱里面,因为它都会以邮件方式进行保存,如果执行的次数过多也会造成垃圾文件。所以都重定向到垃圾箱内比较好(当然 at命令可以不用这样设置,因为它仅仅执行一次,甚至用-m 强行发送邮件判断它是否执行)。

    • 也可以用一个脚本删除多少天以前的邮件或者log文件来清除此缓存。
    • 注意它的log文件只是检测了这个计划任务是否执行,而不检测里面的命令的输出信息,里面额命令的输出信息才是保存在邮件中的信息。
  4. 脚本或者命令最好写全路径,如果不写的话注意要定义好PATH变量(要么利用crontab -ekey = value方式,要么就是利用source命令;具体看crontab另一篇博客。

  5. 注意一点,如果时间格式如下所示,则代表的日子和星期并非是与的关系,而是或的关系,它表示每月的1 10 20日每周的星期1 5 都会进行计划任务

    • 如果非要是与的关系,则只能在后面的CMD中进行判断了,比如下面的判断是否是周日
    30 23 1,10,20 * 1,5 CMD 30 23 1,10,20 * * [ `date +%w` -eq 0 ] && cmd

8. 举例:

8.1. 例1:计划任务来重启httpd服务

cat /etc/crontab crontab -e * * * * * /usr/bin/killall -0 httpd &> /dev/null || systemctl start httpd crontab -l cat /var/spool/cron/root cat /var/log/cron

8.2. 思考:

如何在秒级别运行任务?

每分钟运行3次命令

* * * * * for min in 0 1 2; do echo "hi"; sleep 20; done

如何实现每7分钟运行一次任务?

  • 因为每7分钟和每小时60分钟的最小公倍数是420,也就是7个小时;同时每天24小时和7个小时的最小公倍数也是168小时,也就是7天。由此可见每7天才有一次大的循环;
  • 因此再不考虑要求特定时间,只要求每过7分钟执行一次任务的话;则如下:
* * * * * (( `date +%s`/60%7 )) || echo "hello"

8.2.1. sleep命令:

sleep NUMBER[SUFFIX]... SUFFIX: s: 秒, 默认 m: 分 h: 小时 d: 天

8.2.2. usleep:标准单位为1us,微秒

usleep 1000000 :此为1秒
最后修改日期: 2021年7月7日

作者

留言

撰写回覆或留言

发布留言必须填写的电子邮件地址不会公开。