Linux定时任务靠crontab(周期)和at(一次性),需掌握时间格式逻辑、环境差异及陷阱:crontab五字段为分时日月周,日与周须同时满足;脚本执行需用绝对路径或设PATH;at需atd服务运行且注意shell兼容性;务必记录日志并检查系统日志。

Linux 中的定时任务调度主要靠 crontab(周期性执行)和 at(一次性执行)实现。掌握它们的关键不是死记语法,而是理解时间格式逻辑、环境差异和常见陷阱。
crontab 时间字段怎么填才不踩坑
crontab 的 5 个时间字段顺序是:分 时 日 月 周,容易把“日”和“周”混淆。注意:日(1–31)和周(0–7,0 和 7 都代表周日)**同时满足才会触发**,不是“或”的关系。
- 想每月 1 号凌晨 2 点运行:
0 2 1 * * - 想每周一早上 9 点运行:
0 9 * * 1(不是 0,因为 0 是周日) - 想每 15 分钟执行一次:
*/15 * * * *(不是 */15/15) - 想在工作日(周一到周五)的每小时整点运行:
0 * * * 1-5
别忘了用 crontab -e 编辑后保存,系统不会自动提示是否生效——改完建议立刻用 crontab -l 确认内容已更新。
脚本在 crontab 里不执行?多半是环境问题
crontab 默认使用极简 shell 环境(PATH 很短,通常不含 /usr/local/bin 或 ~/bin),且不读取用户 .bashrc/.profile。所以即使命令在终端能跑,放进 crontab 就报 “command not found”。
- 写绝对路径:比如用
/usr/bin/python3而不是python3 - 显式设置环境变量:在 crontab 文件顶部加一行
PATH=/usr/local/bin:/usr/bin:/bin - 或者统一用 bash 包装:
0 3 * * * /bin/bash -c 'cd /home/user/script && ./backup.sh >> /tmp/backup.log 2>&1'
务必重定向输出(>> 和 2>&1),否则错误信息会静默丢失,排查无从下手。
at 命令适合什么场景?怎么确保它真能跑
at 适用于明确时间点的一次性任务,比如“今晚 23:45 关机”、“明天上午 10 点发邮件提醒”。它不依赖用户登录状态,但需要 atd 服务开启。
- 检查 atd 是否运行:
systemctl is-active atd,如为 inactive,运行sudo systemctl enable --now atd - 提交任务:
echo "shutdown -h now" | at 23:45;也可用at now + 10 minutes - 查看待执行任务:
atq;取消任务:atrm
注意:at 默认使用 /bin/sh,如果你的命令依赖 bash 特性(如数组、[[ ]]),得显式指定:echo '#!/bin/bash\necho $(date)' | at now,并确保第一行是 shebang。
调试与日志:别让定时任务变成黑盒
crontab 和 at 都不会主动告诉你失败原因。开启基础监控很关键:
- 所有定时任务末尾加日志记录,例如:
2>&1 | tee -a /var/log/myjob.log - 检查系统级 cron 日志:
sudo grep CRON /var/log/syslog(Ubuntu/Debian)或/var/log/cron(CentOS/RHEL) - 临时测试可用
run-parts --test /etc/cron.hourly检查脚本命名和权限(必须可执行、无扩展名、不以 . 开头) - 用
date -d "next Friday"快速算出相对时间表达式对应的具体时刻,避免手动推算出错
不复杂但容易忽略。









